Aceste instrucțiuni au fost introduse înainte de separarea excepțiilor de întrerupere și excepții și sunt aproape întotdeauna denumite instrucțiuni de întrerupere Deoarece manevrele de întrerupere și excepții DOS de obicei nu fac distincție între mecanismul de apelare, comanda INT poate transfera controlul atât către manipulatorii de întrerupere și excepții ! Trucuri complexe de programare După cum se arată în Capitolul , întreruperile software, adică transferul de control folosind instrucțiunea INT, sunt mijlocul principal de apelare a unei proceduri DOS și BIOS, deoarece, spre deosebire de apelarea prin instrucțiunea CALL, nu trebuie să cunoașteți adresa a procedurii care se apelează aici - este suficient doar numărul Pe cealaltă parte a interfeței, luați în considerare modul în care este construit un handler de întrerupere software Managerii de întrerupere Când o instrucțiune INT este executată în modul real, controlul este transferat la o adresă care este citită dintr-o matrice specială, tabelul vector de întrerupere, începând din memorie la adresa OOOOh:OOOOh Fiecare element al acestui tablou reprezintă adresa îndepărtată a handler-ului de întrerupere sub forma segmentxlocation, sau octeți nuli dacă nu este setat niciun handler Instrucțiunea INT împinge registrul flag și adresa de returnare departe în stivă, deci} comanda iret, care este complet analog cu ele în modul real ; Un exemplu de gestionare de întreruperi software int handler proc departe mov ax,O iret int handler/endp După ce handlerul este scris, următorul pas este să îl legați la numărul de întrerupere selectat Acest lucru se poate face prin scrierea directă a adresei sale în tabelul vector de întrerupere, astfel: împinge O pop es pushf-cli ;' manipulator ; Adresa segmentului tabelului vectorial de întrerupere - ; în ES ; Împingeți registrul steagului pe stivă ; Dezactivați întreruperile (ca să nu se întâmple ; întrerupere hardware între următoarele comenzi, care teoretic poate provoca INT h în acest moment ; când offset-ul a fost deja scris, dar adresa segmentului nu a fost încă scrisă, ; care va transfera controlul într-o zonă nedefinită de memorie) ; Pune adresa de departe a handler-ului int handler ; la tabelul vector de întrerupere, la elementul numărul ti ; (una dintre întreruperile neutilizate) mov word ptr es:[ h* ], offset int handler mov word ptr es:[ h* + ], seg int handler popf ; Restabiliți valoarea inițială a indicatorului IF Acum comanda INT h va apela handlerul nostru, adică va duce la scrierea în registrul AX Înainte de a ieși, programul trebuie să restabilească toți gestionatorii de întreruperi vechi, chiar dacă erau întreruperi neutilizate, cum ar fi h - autorul unui alt program ar putea gândi la fel Pentru a face acest lucru, trebuie să salvăm adresa vechiului handler înainte de fragmentul de cod anterior, astfel încât setul complet de acțiuni pentru programul care interceptează întreruperea h va arăta astfel: Capcanarea întrerupe eu eu ■і împinge Despre pop ■ es ; Copiați adresa handler-ului anterior în variabila oldjiandler ; mov - mov eax,dword ptr es:[ h* ] dword ptr oldjiandler, eax Instalați manipulatorul nostru, pushf cli mov mov ; popf word ptr es:[ h* ], offset intjiandler word ,ptr es:[ h* + ], seg intjiandler Corpul' programului ! [ ] ■ Restaurați handlerul anterior ■ push pop pushf ■ cli mov mov popf es eax,word ptr old handler word ptr es:[ h* ],eax Deși modificarea directă a tabelului de vectori de întrerupere pare convenabilă, aceasta nu este totuși cea mai bună abordare pentru a seta un handler de întreruperi și ar trebui utilizată numai în cazuri excepționale, cum ar fi în interiorul gestionarilor de întreruperi Pentru programele obișnuite, DOS oferă două funcții de sistem: h și h - setați și citiți adresa operatorului de întrerupere, care sunt recomandate pentru utilizare în condiții normale: ; Copiați adresa handler-ului anterior în variabila old handler mov int ax, h ; AH - h, AL = numărul de întreruperi h; Funcția DOS: citire; adresa operatorului de întrerupere mov mov cuvânt ptr oldjiandler, bx ; Retur offset în cuvântul BX ptr old handler+ ,es ; și adresa segmentului în ES ; Instalați handlerul nostru mov mov mov mov int ax, h ; AH = h, AL = numărul de întreruperi, dx seg intjiandler ; Adresa segmentului - ds,dx ; în DS, / dx,offset int handler ; offset în DX h; Funcția DOS: instalare; handler în corpul programului; (nu uitați că ES ; s-a schimbat de la apelul funcției de de ore!) [• ; Restaurați handlerul anterior '■ Id-uri mov int dx,old handler ; Segmentează adresa în DS și offset în DX ax, h ; AH = h, AL = numărul de întreruperi h; Instalați handler ; Trucuri complexe de programare În mod obișnuit, gestionatorii de întreruperi sunt utilizați pentru a gestiona întreruperile de la dispozitive externe sau pentru a solicita servicii de la alte programe Aceste posibilități sunt discutate mai târziu, dar aici este un exemplu de utilizare a unui handler de întrerupere normal (sau, în acest caz, o excepție de eroare) pentru a găsi rapid minimul și maximul într-o gamă largă de date ; procedura jsinmax ; Găsește valorile minime și maxime într-o matrice de cuvinte ; Intrare: DS:IN = adresa de la începutul matricei ; CX = numărul de elemente din matrice ; Ieșire: ; AX = element maxim; BX = element minim minmax proc aproape ; Instalați handlerul nostru de întrerupere apăsați Pop es mov eax,dword ptr es:[ * ] mov dword ptr old int ,eax mov word ptr es:[ * ],offset int handler mov word ptr es:[ * ]+ ,cs Inițializați minimul și maximul ca prim element al matricei mov ax,word ptr [bx] mov cuvânt ptr limita inferioară,ax mov word ptr upper bound,ax ; Procesați matricea mov di, ; Începeți cu al doilea element bverifica: mov ax,word ptr [bx][di] ; Citiți elementul în AX boundax limite ; Comanda BOUNO apelează ; excepție - eroarea , ; dacă AX nu este în ; limită inferioară/limită superioară adaugă di, ; elementul următor bucla bverificare; Buclă prin toate elementele Restaurați handlerul anterior, mov eax,dword ptr old int mov dword ptr es:[ * ],eax Returnează rezultatele mov ax,word ptr 'upper bound mov bx,word ptr low bound ret limite: limita inferioară dw ? upper bound dw ? old int dd ? INT handler pentru procedura minmax Comparați AX cu upper bound și lower bound și copiați AX într-una dintre ele Handler-ul nu gestionează conflictele între Capcanarea întrerupe ITT ii I ; excepție"! SUNET și software-ul întrerup ecranul de imprimare INT ; Apăsarea tastei PrtScr în timp ce se execută procedura mirmax va avea ca rezultat ; la o eroare Pentru a remedia acest lucru * puteți, de exemplu, să verificați octetul de inchi, ; către care se adresează adresa de retur, dacă este OCDh ; (codul de comandă INT), atunci handlerul bcl este numit ca 'IT * int handler proc far cmp ax,word ptr limita inferioară ; Comparați AX cu limita inferioară, jl its lower ; Daca nu mai putin - ; a fost o încălcare mov word ptr upper bound,ax ; marginea de sus, iret' its lower: ' / mov word ptr limita inferioară,ax ; Altfel a fost o încălcare a iret ; marginea de jos int handler endp miptah endp Desigur, aruncarea unei excepții în caz de eroare durează mult timp, dar dacă matricea este suficient de mare și neordonată, o parte semnificativă a verificărilor vor fi fără erori și rapide De asemenea, puteți gestiona alte situații speciale cu propriii gestionați de excepții, cum ar fi gestionarea Division by Zero și alte excepții care apar în Program În modul real, există doar șase excepții posibile: □ #DE (împărțire la zero) - INT - o eroare care apare la depășire și împărțire la zero Ca și în cazul oricărei erori, adresa de retur indică o comandă de eroare; □ #DB (trace interrupt) - INT - capcană care apare după ce fiecare comandă este executată dacă flag-ul TF este setat la Folosit de depanatorii care operează în mod real; □ #OF (overflow) - INT - o capcană care apare după executarea comenzii INTO dacă este setat flag-ul OF; □ #BR (overflow la BOUND) - INT - eroarea pe care am considerat-o deja, care apare la executarea comenzii BOUND; □ #UD (comandă invalidă) - INT - o eroare care apare la încercarea de a executa o comandă care nu este disponibilă pe acest procesor; □ #NM (fără coprocesor) - INT - o eroare care apare când se încearcă executarea unei comenzi FPU dacă nu există FPU Întreruperi de la dispozitive externe Întreruperile de la dispozitive externe, sau întreruperi hardware, este ceea ce se înțelege prin termenul „întrerupere” Dispozitivele externe (tastatură, unitate de disc, cronometru, placă de sunet etc ) dau un semnal prin care procesorul întrerupe execuția programului și transferă controlul către manipulatorul de întreruperi În total, pe computerele personale sunt folosite întreruperi hardware, deși teoretic capacitățile arhitecturii fac posibilă creșterea numărului acestora la Trucuri complexe de programare eu Să le analizăm pe scurt, în ordinea descrescătoare a priorității („întreruperea are o prioritate mai mare” înseamnă că până la finalizarea responsabilului său, întreruperile cu priorități scăzute vor aștepta rândul lor): □ IRQO (INT ) - întreruperea temporizatorului de sistem, apelată de , ori pe secundă Handler-ul implicit pentru această întrerupere apelează INT ICh de fiecare dată când este apelat, deci dacă programul trebuie doar să primească control regulat și să nu reprogrameze cronometrul, se recomandă utilizarea întreruperii ICh; □ IRQ (INT ) - întrerupere de la tastatură, declanșată de fiecare dată când o tastă este apăsată și eliberată de pe tastatură Handler-ul standard pentru această întrerupere realizează destul de multe funcții, de la repornirea cu Ctrl-Alt-Del până la plasarea codului tastei în buffer-ul tastaturii BIOS; □TRQ - întreruperile hardware IRQ - IRQ sunt conectate la această intrare pe primul controler de întrerupere, dar multe BIOS-uri redirecționează IRQ către INT OAh; □ IRQ (INT h) - întrerupere a ceasului în timp real, apelată de ceasul în timp real când alarma se declanșează și dacă este setată să genereze o întrerupere periodică (în acest din urmă caz, IRQ este apelat de de ori pe secundă); □ IRQ (INT OAh sau INT h) - întreruperea fasciculului invers, apelată de unele adaptoare video în timpul traseului fasciculului invers Folosit adesea de dispozitive suplimentare (de exemplu, plăci de sunet, adaptoare SCSI etc ); □ IRQ (INT h) - utilizat de dispozitive suplimentare; □ IRQ (INT h) - utilizat de dispozitive suplimentare; □ IRQ (INT b) - mouse pe sisteme PS, utilizat de dispozitive suplimentare; □ IRQ (INT h sau INT h) - eroare coprocesor matematic În mod implicit, această întrerupere este dezactivată atât pe FPU, cât și pe controlerul de întrerupere; □ IRQ (INT h) - întreruperea primului controler IDE „operațiune finalizată”; □ IRQ (INT h) - întreruperea celui de-al doilea controler IDE „operațiune finalizată”; □ IRQ (INT OBh) - întrerupere port serial COM , declanșată dacă portul COM a primit date; □ IRQ (AXA INT) - întreruperea portului serial COM , declanșată dacă portul COM a primit date; □ IRQ (INT ODh) - întrerupere LPT , utilizată de dispozitive suplimentare; □ IRQ (INT OEh) - întreruperea unității de disc „operațiune finalizată”; □ IRQ (INT OFh) - întrerupere LPT , utilizată de dispozitive suplimentare Capcanarea întrerupe Întreruperile hardware utile pentru programe sunt întreruperi ale temporizatorului și tastaturii sistemului Deoarece manipulatorii de întreruperi standard pentru aceste întreruperi îndeplinesc multe funcții de care depinde sistemul, ele nu pot fi înlocuite complet, așa cum am procedat noi cu handlerul INT Trebuie să apelați handler-ul Previous, transmițându-i controlul după cum urmează (dacă adresa sa este stocate în variabila old handler - vezi mai jos) exemple înainte): pushf caii old handler Aceste comenzi efectuează aceeași acțiune ca și comanda INT; (salvați steaguri pe stivă și transferați controlul ca comanda caii), așa că atunci când handlerul se termină cu comanda IRET, controlul revine la programul nostru Este atât de convenabil să apelați manerul anterior la începutul propriei dvs O altă modalitate este o comandă jmp simplă: jmp cs:old handler conduce la tbmu că la executarea instrucțiunii IRET de către vechiul handler, controlul va trece imediat la programul întrerupt Această metodă este folosită dacă este necesar ca noul handler să funcționeze mai întâi și apoi transferă controlul celui vechi În exemplul următor, să vedem cum este interceptată întreruperea temporizatorului: ; temporizator asm ; Demonstrarea interceptării unei întreruperi de cronometru de sistem: afișarea orei curente ; în colțul din stânga ecranului modelul minuscul cod ; Pentru pusha/popa și ture org h începe proc aproape ; Salvați adresa gestionarului de întrerupere Ch anterior movax, Ch ; AH = h, AL = numărul de întreruperi int h; Funcția DOS: determina adresa handlerului mov word ptr old int Ch,bx ; întrerupe mov word ptr old int Ch+ ,es ; (revine la ES:BX) Instalați handlerul nostru movax, Ch ; AH = h, AL = numărul de întreruperi mov dx offset int Ch handler ; OS:DX - adresa handlerului int Wi ; Setați gestionarea întreruperilor ; Aici se află programul* real, cum ar fi apelarea command com muta ah, int h'; Se așteaptă apăsarea oricărei taste ; Sfârșitul programului ; Restaurați manerul de întrerupere Ich anterior movax, Ch ; AH = h, AL = numărul de întrerupere, mov dx word ptr old int Ch+ Trucuri complexe programate* mov mov int ds dx dx word ptr cs:old int Ch ; DS:DX - adresa handlerului h X ret old int Chdd ? ; Adresa handler-ului anterior este stocată aici sart position dw ; Poziția pe ecran de afișat ; ora curentă start endp Managerul de întrerupere СІ Afișează ora curentă la start position pe ecran (numai în modul text int Ch handler proc departe pusha; Gestionar întreruperi hardware împinge; trebuie să păstreze TOATE registrele împinge ds apăsare CS ; Doar la intrarea în handler pop ds ; valoarea registrului CS - mov ah,O h ; Funcția h întrerupere Ah: int Ah ; Timp de citire de la RTC jc- exit handler ; Dacă ceasul este ocupat - altă dată ; AL = oră în format BCD cai bcd asc ; Convertiți în ASCII mov byte ptr output line[ ], ah ; Pune-le înăuntru mov byte ptr output line[ ],al ; line output line ' mov al,cl ; CL = minut în format GCO cai bcd asc- mov byte ptr output line[ ],ah mov byte ptr output line[ ], al mov al, dh / ; DH = secundă în format BCD caii bcd asc mov byte ptr output line[ ],ah mov byte ptr output line[ ],al mov cx output line l ; Numărul de octeți dintr-un șir este în CX apăsați OB h papi; Adresă în memoria video - mov di,word ptr start position ; în ES:DI mov si offset output line ; Adresă de linie în DS:SI cld! rep movsb * ; Copiați linia exit handler: pop ds ; Restaurați toate registrele pop es popa jmp cs:old int Ch ; Transferați controlul operatorului anterior Procedura bcd asc Convertește cifra înaltă a unui număr BCD împachetat din AL într-un caracter ASCII, Capcanarea întrerupe I ȚȚ Ț■ ; care va fi plasat în AH iar cifra inferioară în caracterul ASCI^ în AL bcd asc proc lângă ' mov ah, al și al, OFh; Lăsați biți la nivel scăzut în AL shrah, ; Schimbați cei biți înalți la AH sau ax, h; Convertiți în caractere ASCII ret bcd^asc endp ; Șirul „OOh : ” cu atributul IFh (alb pe albastru) după fiecare caracter output line db ' ', Fh,' ', Fh,' ', Fh,'h', Fh db ', Fh, ' ', Fh, ' ', IFh, ': ', IFh db ' ', IFh,' ',IFh, '', Fh output line l equ $-output line inIch handler endp • sfârşitul începutului Dacă în acest exemplu, în loc să așteptați o apăsare a tastei, să puneți un program care funcționează în modul text, cum ar fi tinyshell din secțiunea , acesta va fi executat ca de obicei, dar ora curentă va fi afișată constant în colțul din dreapta sus, adică un astfel de program va face două lucruri simultan Exact pentru asta este folosit mecanismul de întrerupere hardware - ele permit procesorului să execute un program, în timp ce programe separate țin evidența timpului, citesc caracterele de la tastatură și le pun într-un buffer, primesc și transmit date prin porturi seriale și paralele și chiar oferă multitasking, schimbând procesorul între diferite sarcini prin întreruperea temporizatorului sistemului Desigur, gestionarea întreruperilor nu ar trebui să dureze mult: dacă o întrerupere are loc suficient de frecvent (de exemplu, o întrerupere a unui port serial poate apărea de de ori pe secundă), handlerul său trebuie neapărat să se execute într-un timp mai scurt Dacă, de exemplu, gestionarea întreruperilor temporizatorului rulează timp de / , secunde, adică jumătate din timpul dintre întreruperi, întregul sistem va funcționa de două ori mai lent Și dacă un alt program cu același handler lung interceptează această întrerupere, sistemul se va opri complet De aceea, manipulatorii de întrerupere sunt de obicei scrise exclusiv în asamblator Reintrare Să presupunem că avem propriul nostru handler de întreruperi software, care este apelat de cei care gestionează două întreruperi hardware și lăsăm aceste întreruperi hardware să apară imediat după cealaltă În acest caz, se poate dovedi că a doua întrerupere hardware va avea loc atunci când execuția handler-ului nostru software nu s-a încheiat încă De cele mai multe ori acest lucru nu va cauza probleme, dar dacă handler-ul accesează orice variabilă din memorie, poate întâmpina blocări „rare” ireproductibile De exemplu, să presupunem că handler-ul are un contor variabil folosit ca numărător care numără de la la : Trucuri complexe de programare mov al,byte ptr^counter f cmp al, contor jb Bine »> aici s-a produs a doua întrerupere sub al, mov byte ptr counter al counter ok: Citiți contorul în AL Verificați dacă există preaplin Dacă contorul ajunge la , scade și salvați contorul Dacă valoarea contorului a fost, de exemplu, și a doua întrerupere a avut loc după verificare, dar înainte de a scădea , al doilea apel către handler va obține aceeași valoare de și o va reduce cu Controlul va reveni apoi și următoarea instrucțiune sub al, va reduce AL din nou cu și va nota numărul rezultat - în loc Dacă se calculează apoi ceva precum adresa de memorie care urmează să fie scrisă din valoarea contorului, este foarte posibil să apară o eroare Un astfel de handler de întrerupere se spune că nu este reintre Pentru a proteja astfel de secțiuni critice de cod, ar trebui să dezactivați temporar întreruperile, ca aceasta: cli ; Dezactivați întreruperile Mov al,byte ptr counter cmp al, jb counter ok sub al, mov byte ptr counter al counter ok: sti; Activați întreruperile Trebuie reținut că, în timp ce întreruperile sunt dezactivate, sistemul / nu urmărește modificările ceasului, nu primește date de la tastatură, așa că întreruperile trebuie activate cât mai curând posibil Este întotdeauna mai bine să reconsiderați algoritmul utilizat și, de exemplu, să stocați variabilele locale pe stivă sau să utilizați o comandă CMPXCHG special concepută, care vă permite să comparați și să scrieți într-o variabilă globală în același timp Din păcate, sub MS DOS, cel mai important handler de întrerupere din sistem, handler-ul INT b, nu este reintrant Spre deosebire de întreruperile BIOS, ai căror handlere folosesc stiva programului întrerupt, gestionarea funcției de sistem DOS scrie în SS:SP adresa de jos a unuia dintre cele trei stive interne DOS Dacă o funcție a fost întreruptă de o întrerupere hardware al cărei handler a numit o altă funcție DOS, va folosi aceeași stivă, suprascriind orice funcție întreruptă pusă acolo Când controlul revine la funcția întreruptă, va exista gunoi pe stivă și va apărea o eroare Cea mai bună cale de ieșire este să nu folosiți deloc întreruperile DOS de la manipulatorii de întreruperi hardware, dar dacă este cu adevărat necesar, luați măsurile de precauție necesare Dacă întreruperea a avut loc în timp ce nu rula nicio funcție de sistem DOS, acestea pot fi utilizate în siguranță Pentru a determina dacă DOS este ocupat sau nu, trebuie mai întâi, înainte de a vă instala propriile handlere, să aflați adresa indicatorului DOS ocupat Capcanarea întrerupe NU! j I Funcția DOS h: Obține adresa flag DOS ocupat -Intrare: AH - h Ieșire: ES:BX - adresa steagului de ocupat DOS de un octet ES:BX - - adresa flagului de eroare critică DOS pe un octet Acum, operatorul de întrerupere poate verifica starea acestor steaguri și, dacă ambele steaguri sunt zero, este permisă utilizarea liberă a funcțiilor DOS Dacă indicatorul de eroare critică nu este zero, nu pot fi utilizate funcții DOS Dacă indicatorul DOS ocupat nu este zero, puteți utiliza doar funcțiile Olh - OCh, iar pentru a utiliza orice altă funcție, va trebui să amânați acțiunile până când DOS este liber Pentru a face acest lucru, trebuie să stocați numărul funcției și parametrii în unele variabile în memorie și setați handlerul de întrerupere h sau ICh Acest handler va verifica steaguri de ocupat la fiecare apel și, dacă DOS este liber, va apela funcția cu numărul și parametrii lăsați în variabilele din memorie În plus, secțiunea programului după verificarea indicatorului de ocupat este critică, iar întreruperile trebuie dezactivate Nu toate funcțiile DOS revin rapid - funcția de citire a tastaturii poate rămâne în această stare timp de minute, ore sau chiar zile până când utilizatorul revine și apasă o tastă, timp în care indicatorul de ocupat DOS este setat la DOS asigură o astfel de situatie Toate funcțiile de introducere a caracterelor de așteptare apelează INT h în același ciclu în care interogează tastatura, așa că dacă setați un handler de întrerupere de h, puteți apela toate funcțiile DOS din acesta, cu excepția - OCh Un exemplu de apel DOS de la un handler de întreruperi de la un dispozitiv extern este discutat mai jos, în programele rezidente Deocamdată, trebuie remarcat faptul că funcțiile BIOS, una dintre care am numit-o în exemplul nostru timer asm, se dovedește adesea a nu fi reintroduse În special, gestionatorii de întreruperi de program , , , OBh, OCh, ODh, OEh, lOh, h, h, h, h diferă în acest sens Deoarece BIOS-ul nu oferă niciun semnal de ocupat, va trebui să creați unul singur: int handler proc departe inc cs:byte ptr int busy ; Creșteți f/g de ocupare pushf ; Transferați controlul la vechi; handler INT h, caii cs:dword ptr old int ; emulând instrucțiunea INT dec cs:byte ptr int busy ; Reduceți steag de ocupat iret int busy db / int handler endp Acum, manipulatorii de întreruperi hardware pot folosi instrucțiunea INT h dacă indicatorul de ocupat intlO ocupat este zero și acest lucru nu va duce la erori decât dacă există un alt handler de întreruperi care va accesa și INT Oh și nu va ști nimic despre semnalul nostru de ocupat Trucuri complexe programate Programe pentru rezidenți Programele care rămân în memorie după ce returnați controlul la DOS sunt numite programe rezidente A face rezident al unui program este la fel de simplu ca apelarea unei funcții speciale de sistem DOS Funcția DOS h: Păstrați programul rezident Intrare: AH = h AL = cod de retur DX = dimensiunea rezidentului în paragrafe de octeți (mai mare de h), numărând de la începutul PSP În plus, o versiune anterioară a acestei funcții, întreruperea h, există și este uneori folosită: INT h-, Concediu/, rezident program Intrare: AH = h DX = adresa ultimului octet al programului (numărând de la începutul PSP) + Această funcție nu poate lăsa rezidente programe mai mari de KB, dar multe programe scrise în assembler respectă această regulă Deoarece programele rezidente reduc cantitatea de memorie principală, ele sunt întotdeauna scrise în limbaj de asamblare și optimizate pentru a atinge dimensiunea minimă Se obișnuiește să se împartă programele rezidente în active și pasive, în funcție de dacă interceptează întreruperile de la dispozitive externe sau primesc control numai dacă programul apelează în mod specific comanda INT cu numărul și parametrii de întrerupere doriti Program de rezidență pasivă Ca prim program rezident, să luăm în considerare rezidentul pasiv, care va fi activat atunci când programele încearcă să apeleze INT h și să interzică ștergerea fișierelor de pe discul specificat ; tsr asm- ; Un exemplu de program rezident pasiv , ; Împiedică pe toată lumea să șterge fișierele de pe unitatea specificată pe linia de comandă ; programe care folosesc instrumente DOS modelul minuscul cod org Ch envseg dw ? ; Adresa de segment a unei copii a mediului DOS org h cmd lendb ? ; Lungimea liniei de comandă cmd line db ? ; Începutul liniei de comandă org h ; Programul COM start: old int h: full spec: mutați și comparați: cmp je pop pop not fn h: popf jmp fn h: irrt h jmp short initialize dw ; Această comandă are octeți, deci ; împreună cu ei obținem ; old int hdd ? handler proc departe ; Gestionarea întreruperii h pushf ; Salvați steaguri cmpah, h ; Dacă funcția h (eliminați Je fn h; fişier) cmp ax, h ; sau h (ștergeți fișierul cu nume lung) je fn h; porniți manevrătorul nostru jmp scurt not fn h ; În caz contrar, controlul transferului; manipulatorul anterior împinge toporul'; Salvare Modificabil push bx ; registre movbx,dx cmp byte ptr ds:[bx+ ],': ' ; Dacă al doilea caracter al șirului ASCIZ, ; a trecut INT h, colon - primul; caracterul trebuie să fie un nume de unitate je full spec mov ah, h; In caz contrar - int h Funcția DOS h - determinați unitatea curentă adăugați al, 'A'; Convertiți numărul discului; la litera mare jmp scurt comparare; Treci la comparație al byte pțr [bx] al, b ; AL = numele unității din șirul ASCIZ ; Convertiți în majuscule al byte ptr cs:cmd line[ ] ; Dacă discurile sunt access denied bx ax ; potrivire - interzice accesul ; În caz contrar, restaurați ; registre dword ptr cs:old int h ; și steaguri ; și controlul transferului ; la manipulatorul anterior INT h access denied: pop pop popf puști mov or bx topor bp bp, sp cuvânt ptr[bp+ ], POP bp ; Restaurați registrele ; Setați steag de transport ; (bit ) în registrul steagului, ; pe care instrucțiunea INT l-a împins pe stivă ; înainte de adresa de retur Legea Trucuri complexe programate) mov ax, ; Returnează codul de eroare „acces ^ refuzat iret; Intoarce-te program int h handler endp initialize proc near cmp byte ptr cmd len, ; Verificați dimensiunea liniei de comandă jne not install ; (ar trebui să fie - spațiu, unitate, două puncte cmp byte ptr cmd line[ ], ': ' ; Verificați al treilea caracter jne not install ; linie de comandă (ar trebui să fie două puncte mov al,byte ptr cmd line[ ] şi al, b; Convertiți secunda; caracter la literă mare cmp ■al,'A' ; Verificați că nu este jb not install ; mai puțin decât „A” și nu mai mult cmp al 'Z'; „Z” ' ja not install ; Dacă cel puțin una dintre aceste condiții; neexecutat - da informatii; despre program și ieșire , eu ; În caz contrar, începeți procedura de inițializare mdv ax, h ; AH = h, AL = numărul de întreruperi • int h ; Obțineți adresa handlerului INT h mov word ptr old int h,bx ; și mănâncă-l în old int h > mov word ptr old int h+ ,es mov ăx, h ; AH = h, AL = numărul de întreruperi mov dx,offset int h handler ; DS:DX este adresa operatorului nostru int h; Instalați handlerul INT h mov ah, h; AH = h mov es word ptr envseg ; ES = bloc adresa segmentului cu ; copie a mediului DOS int h; Eliberați memoria din mediu mov dx, offset initialize; DX - adresa primului octet după sfârșit; parte rezidentă a programului int h; Încheiați execuția cu ; rezident not install: mov ah, ; AH = h mov dx, utilizare offset; DS:DX = adresa liniei cu informații despre ; folosind programul int h; Afișarea unui șir pe ecran ret; Încheierea normală a programului ; Textul pe care programul îl produce atunci când rulează cu o linie de comandă nevalidă: usage db „Utilizare: tsr com D:”,ODh,OAh db „Interzice ștergerea pe unitatea D:”,ODh,OAh db ■'$" inițializați endp sfârşitul începutului Programe pentru rezidenți Dacă rulați acest program din linia de comandă D:, niciun fișier de pe unitatea D nu poate fi șters cu comanda Del, instrumente shell precum Norton Commander și majoritatea programelor DOS Această interdicție, totuși, nu se va aplica pentru Far shell, care utilizează funcțiile sistemului Windows API și pentru programe precum Disk Editor, care accesează discuri folosind funcții BIOS (INT IB) Chiar dacă am eliberat memoria ocupată de mediul DOS (care ar fi putut fi în plus sau chiar de octeți), programul nostru încă ocupă de octeți de memorie deoarece primii de octeți sunt rezervați blocului PSP Este posibil să lăsați rezidentul programului fără PSP - pentru aceasta, partea de instalare a programului trebuie să copieze partea rezidentă folosind, de exemplu, mutări la începutul PSP Dar acest lucru ridică mai multe probleme simultan: în primul rând, comanda INT h, ca și funcția DOS h, utilizează datele de la PSP pentru lucrul său; în al doilea rând, codul pentru partea rezidentă trebuie să fie scris pentru a funcționa de la zero offset, și nu din IOOB, ca de obicei; și în al treilea rând, unele programe care examinează blocurile de memorie alocate determină sfârșitul blocului la adresa situată în PSP-ul programului proprietar al blocului la offset Prima problemă poate fi rezolvată manual prin crearea de blocuri de memorie separate pentru rezident și instalare părți ale programului, noul PSP Pentru partea de instalare și încheierea programului cu funcția obișnuită Cb sau INT b Există programe reale care fac acest lucru (de exemplu, programul de suport pentru formatul de disc personalizat PU ), dar nu vom complica prea mult primul nostru exemplu și vom copia partea rezidentă nu în poziția , ci în poziția b, adică începând de la mijloc al PSP-ului, lăsându-l să conțină toate valorile necesare pentru funcționarea normală a funcțiilor DOS Înainte de a face acest lucru, rețineți că atât numărul discului, cât și adresa manipulatorul INT b se schimbă numai atunci când rezidentul este instalat și sunt constante pe toată durata funcționării sale Mai mult, fiecare dintre aceste numere este folosit o singură dată În astfel de condiții, se dovedește că puteți introduce numărul discului și adresa tranziției la vechiul handler direct în codul programului În plus, după aceea, rezidentul nostru nu se va mai referi la niciun variabile cu adrese specifice, ceea ce înseamnă că codul său devine relocabil, adică poate fi executat prin copierea lui în orice zonă de memorie ; tsrpsp asm ; Un exemplu de program rezident pasiv cu portarea codului la PSP ; Previne ștergerea fișierelor de pe unitatea specificată pe linia de comandă, ; toate programele care folosesc facilitatile DOS modelul minuscul cod org Ch envseg dw ? ; Adresa de segment a unei copii a mediului DOS org h cmd lendb ? ; Lungimea liniei de comandă cmd line db ? ; Începutul liniei de comandă org h ; Programul COM i ІІІІ Trucuri complexe programate! start: old lnt h: jmp short initialize; Salt la partea de inițializare lnt h handler proc far ; Gestionarea întreruperii h pushf ; Salvați steaguri cmp ah, h; Dacă funcția h a fost apelată; (sterge un fisier) je fn h cmp ax, h ; sau h (ștergeți fișierul; cu nume lung), je; fn h / porniți handlerul nostru jmp scurt not fn h ; În caz contrar, trimite ; control la manipulatorul anterior fn h: topor de împingere; ; Salvare Modificabil push bx ; registre movbx,dx; Adresarea [edx+ ] ar fi putut fi folosită, dar cuvântul înalt al EDX nu trebuie să fie deloc cmp byte ptr[bx+ ],' ; Dacă al doilea caracter al șirului ASCIZ, * trecut la INT h, două puncte, primul caracter trebuie să fie un nume de unitate je full spec mov-ah, h; In caz contrar: int h; Funcția DOS h - determinați unitatea curentă adauga al 'A'; Convertiți numărul unității în majuscule jmp scurt comparare; Treci la comparație spec complete; mov al,byte ptr [bx] ; al, b; AL = numele unității din șirul ASCIZ Convertiți în majuscule și compara: db Ch ; Începutul codului de comandă CMP AL, număr drive litera: db 'Z' ; Iată procedura de inițializare POP bx ■ introduce litera dorită Aceste registre nu mai sunt Va fi nevoie de toporul POP Dacă discurile se potrivesc - je access denied ; interzice accesul not fn h: Popf Restaurați steaguri și trimiteți db OEah ; control la manipulatorul anterior INT h:; Începutul codului de comandă old int h dd ; JMP, număr FAR Iată procedura de inițializare va scrie adresa handler-ului anterior INT h Programe pentru rezidenți I P P hei access denied: ; dacă este în desfășurare o întrerupere INT bufferlseg dw : I Segmentează adresa tamponului pentru fișier ems handle dw j EMS handle filespec db 'scrgrb bmp' , ; Nume de fișier ; INT Dh handler hw reset D:retf int Dh handler proc departe jmp short actual int h handler ; Omiteți ISP h Trucuri complexe de programare old int Dh jmp despre bh Oh hw reset D dup ( ) dd dw db scurt db actual int h handler: db db its us dword ptr cs:old int Dh mux id je jmp it us: cmp jae cbw mov shl j»P jumptable h, OFCH? al, int nr dl, ax di, cuvânt dw dw dw ; mânerul propriu-zis ; Începutul comenzii CMP AH, număr, ; ID program ; Dacă sună cu ANul altcuiva - nu suntem noi ; Funcții AMIS h și mai sus; nu sunt suportate ; AX = numărul funcției ; DI = numărul caracteristicii ; x din moment ce poate fi sărit ptr cs:jumptable[di] ; Salt la handler offset int D , offset int D no offset int D , offset int D no offset int D , offset int D - tabel de cuvinte, funcții int D : ' ; Verificare disponibilitate Acest ns/mer este ocupat mov al,OFFh ; mov cx, h ; Numărul versiunii programului este împinge cs pop dx X: І - adresa semnăturii AMIS ilfov di,offset amis sign iret int D no: t Funcție neacceptată mov ai,OOh ; Funcția nu este acceptată iret unload failed: ; Controlul este transferat aici dacă cel puțin unul dintre vectori ; întreruperea a fost interceptată de cineva după noi mov al, h; Încărcarea programului a eșuat iret int D : Descărcați programul din memorie cli Sectiune critica push ' poptis; DS - adresa segmentului tabelele vectoriale de întrerupere mov ax,cs ; Adresa segmentului nostru ; Verificați dacă toate întreruperile prinse mai indică către noi ; De obicei, este suficient să verificați numai adresele segmentelor (DOS nu va încărca alta program cu adresa segmentului nostru) str ax,word ptr ds:[ h* + ] jne unload failed str ax,word ptr ds:[ h* + ] jne descărcare eşuat str ax,word ptr ds:[ h* + ] Programe pentru rezidenți jne cmp jne cmp jne unload failed ax,word ptr ds:[ h* + ] unload failed ax,word ptr ds:[ Dh* + ] unload failed push push bx ; Adresă de retur -pe stivă, dx ; Restaurează ID-urile mov int ID-urile mov int ID-urile mov ' if>t mov Id-urile int ID-urile mov int vechii manere de întrerupere ax, h dx,dword ptr cs:old int h h ax, h dx,dword ptr cs:old int h h ' ' ax, h dx,dword ptr cs:old ifltO h h dx,d sword hr dx, ■ ax :pld int h h ax, Dh dx,dword ptr cs:old int Dh h mov cmp je mov int jmp dx word ptr cs:ems handle ; Dacă utilizați EMS dx, no ems to unhook ax, h ; Funcția EMS h: h; memorie alocată gratuită, scurt ems unhooked no ems to unhook: bms unhooked: ; De fapt, descărcarea rezidentului mov int ah, h '; Funcția DOS h: h ; obțineți adresa de segment a PSP-ului întrerupt; proces (în acest caz, PSP - o copie a programului nostru - lansat cu tasta / și) - mov word ptr cs:[ h],bx-, ; Pune-l pe câmp; „adresa segmentului strămoșului” în PSP-ul nostru pop pop , mov mov dx ; Restabiliți adresa de retur din stiva bx word ptr cs:[ Ch,l,dx ; şi pune-l în câmpul cuvânt ptr- cs:[ Ah],bx ; „săriți adresa la ; sfârșitul programului” în PSP-ul nostru z push pop mov int CS ■ bx ; BX = adresa segmentului nostru PSP ah, h; Funcția DOS h: h ; instalați PSP actual Trucuri complexe de programare ; Acum DOS tratează rezidentul nostru ca pe programul curent și scrgrb com /și ca pe apelant; procesul său, căruia îi va transfera controlul după apelarea următoarei funcții mov int ax, CFFh ; Funcția DOS Ch: h ; termina programul int D : ; Obțineți o listă de interceptați; întrerupe mov mov iret int D : mov mov mov iret int Dh handler dx,cs ; Lista în OX:VX bx,offset amis hooklist ; Obțineți o listă de taste rapide al OFFh ; Funcția este acceptată dx cs ; Lista în OX:VX bx, offset amisjiotkeys endp AMIS: Semnătura pentru un program rezident amis sign db „Cubbi ” ; octeți, db „ScrnGrab” ; octeți db „Captură de ecran simplu folosind EMS”, ; AMIS: Amis hooklist of hooked interrupts db h dw offset int h handler db h dw offset int h handler db h dw offset int h handler db Dh dw offset int Dh handler ; AMIS: Lista de taste rapide amis hotkeys db db db ; Cod de scanare cheie (G) dw h ; Semnale de tastatură necesare, dw db Sfârșitul părții rezidențiale Începutul procedurii de Shmrayizare initialize Jmp proc near short initialize entry point ; Sari peste diverse; opțiuni de ieșire fără a instala un rezident, plasate aici ; deoarece le sunt transmise comenzile condiționalului; tranziții cu un interval scurt ieșire cu mesaj nw int ret ab, ; Funcția de afișare a unui șir pe ecran h; Ieșiți din program deja încărcat: ; Dacă programul este deja încărcat în memorie cmp je mov jmp byte ptr descărcare, ; Dacă nu am fost chemați cu / și/ do unload dx offset already msfl' short exit with message no more mux: ; Dacă nu este găsit un identificator liber INT h mov jmp dx offset nb more mux msg scurt exit with message cant unload : mov jmp ; Dacă programul nu poate fi descărcat, dx offset cant unload ițsg short exit withjnessage face descărcare: ; Descărcarea unui rezident: la trecerea controlului aici, AH conține ; ID program - inc mov mov mov int ah al, h ; Funcția de descărcare a rezidenților AMIS dx cs ; Adresă de retur bx/offset exitjjoint ; în DX:BX h; Apelarea rezidentului nostru printr-un multiplexor push cs ; Dacă conducerea a venit aici; nu a fost nicio descărcare pop mov ' jmp ds dx offset cant unload msg short exit with message punct ieșire: ; Dacă controlul a venit aici - s-a produs descărcarea; push pop mov push jmp cs ds dx offset unloaded msg ; Pentru ca comanda RET să iasă la lucru, scurt exit with message punctul de intrare initializare: ; Controlul este transferat aici chiar de la început', cld cmp jne cmp jne mov not unload: byte ptr cmd line( ],'/' not unload byte ptr cmd line[ ], 'u' ; Dacă este apelat cu /u, not unload ■ byte ptr unloading, ; descărcați rezidentul mov mov int ah, dx utilizare offset ; Ieșiți un șir cu informații despre program h mov morejnux: • ah,- ; Scanare de la FFh la OTh mov al,OOh ; Funcția AMIS OOh - verificarea prezenței; rezident ; Trucuri complexe de programare int Dh ; Întreruperea multiplexorului cmp al OOh ; Dacă identificatorul este liber, jne nu este gratuit > mov byte ptr mux id,ah ; introduceți-l direct în codul de gestionare Jmp scurt nextjnux nu este gratis: mov ■ es dx ; În caz contrar - ES:DI = adresa semnăturii AMIS ; programul de apelare mov si,offset amis sign ; DS:SI = adresa semnăturii noastre mov cx, ; Comparați primii octeți gere cmpsb jcxz deja încărcat ; Dacă nu fac față următorul mux: decah; treceți la următorul ID jnz mai mult mux ; Daca este free mux found: cmp byte ptr descărcare, ; și dacă am fi chemați să descarcăm, je cpnt unloadl ; și am venit aici - nu există program; in minte cmp byte ptr mux id O ; Dacă în același timp mux id este încă , je no more mux ; identificatorii au dispărut Verificarea prezenței dispozitivului EMMXXXXO mov dx,offset ems î driver mov topor, D h int h; Deschideți fișierul/dispozitivul jc no emmx mov bx,ax mov ax, h • int h; IOCTL: Obțineți starea fișierului/dispozitivului jc nr ems test dx ewi ; Dacă bitul înalt al DX = , EMMXXXXO este un fișier jz nr ems Alocați memorie pentru buffer în EMS movax, h ; Funcția EMS h: int h; obțineți adresa ferestrei EMS mov bp, bx ; Salvați-l pentru moment în VR movax, h ; Funcția EMS h: mov bx, ;• Avem nevoie de x KB int h; Alocați memorie EMS (identificator în DX) cmpah, ; Dacă apare o eroare (memorie lipsită?) jnz ems failed ; Nu vom folosi EMS mov cuvânt ptr ems mâner dx ; În caz contrar: salvați identificatorul; pentru un rezident movax, h ; Funcție h - afișaj ; Pagini EMS în fereastră mov bx, int h • ; Pagina mov ax, h '- Programe pentru rezidenți unsprezece inc bx int h; Pagina movax, h inc bx int h Pagina ”* mov ax, O h inc bx int h; Pagina muta ah, mov dx,offset emsjnsg; Afișează mesajul de instalare în EMS int h mov ax, bp jmp scurt ems used ems failed: no ems: I Dacă nu există EMS sau nu funcționează, mov ah, Eh int h; Închideți fișierul/dispozitivul EMMXXXXHO no emmx: ; Ocupă memoria partajată muta ah, mov dx,offset convjnsg ; Trimiteți un mesaj despre asta , int h mov sp,lungimea programului+ h+ h ; Mută stiva mov ah, Ah ; Funcția DOS Ah următorul segment = lungimea programului+ C h+ O h+OFh următorul segment =■ următorul segment/ ; O astfel de intrare este necesară doar pentru WASM, restul asamblatorilor ar putea să o scrie într-o singură linie mov bx, următorul segment ; Reduceți memoria utilizată, lăsând lungimea curentă a programului nostru + h pe PSP + h pe stivă int h mov ah, h; Funcția h - alocare memorie bfsize p = bfsize+OFh bfsize p = bfsize p/ movbx,bfsize p ; Dimensiunea fișierului BMP x x pe octeți int ih ; paragrafe ems used: mov word ptr buffer seg,ax ; Stocați adresa tampon pentru rezident ; Copiați antetul fișierului BMP la începutul bufferului mov cx, lungime header BMP mov si,offset BMP header mov di, mutare, ax rep movsb Legea Trucuri complexe de programare Obțineți adresele semnalizatorului de ocupat DOS și ale semnalului de eroare critică (presupunând că versiunea DOS este mai veche decât ) mov ah, h; Funcția h - steag ocupat int h dec bx țș ; Decrementează adresa cu , astfel încât să indice ; la indicatorul de eroare critică, mov cuvânt ptr în dos addr,bx mov cuvânt ptr' in dos addr+ ,es ; și păstrați-l pentru rezident Interceptarea întreruperilor movax, Dh ; AH = h, AL = numărul de întreruperi int h; Obțineți adresa handlerului INT Dh mov word ptr old int Dh,bx ; și puneți-l în old int Dh mov word ptr old int Dh+ ,es movax, h; AH = h, AL = numărul de întreruperi int h; Obțineți adresa handlerului INT h mov word ptr old int h,bx ; și puneți-l în old int h mov word ptr old int h+ ,es mov ax, h ; AH = h, AL = numărul de întreruperi int h; Obțineți adresa handlerului INT h mov word ptr old int h, bx ; și pune-l în old int h mov word ptr old int h+ , es mov ax, h ; AH = h, AL = numărul de întreruperi inf h; Obțineți adresa handlerului INT h mov word ptr old int b,bx ; și pune-l în old int h mov word ptr old int h+ ,es mov,ax, b ; AH = h, AL = numărul de întrerupere int h; Obțineți adresa handlerului INT h mov cuvânt ptr old int h, bx ; și pune-l în old int h mov word ptr old int h+ ,es movax, Dh ; AH = h, AL = numărul de întreruperi mov dx offset int Dh handler ; DS:DX - adresa handlerului int h; Instalați noul handler INT Db movax, h; AH = h, AL = numărul de întreruperi mov dx offset int h handler ; DS:DX - adresa handlerului int h; Instalați un nou handler INT h movax, h ; AH = P, AL = numărul de întrerupere mov dx offset int h handler ; DS:DX - adresa handler- int h,; Instalați un nou handler INT h movax, h; AH = h, AL = numărul de întreruperi mov dx offset int h handler ; DS:DX - adresa handlerului int h; Instalați un nou handler INT h movax, h; AH = h, AL = numărul de întreruperi mov dx offset int h handler ; DS:DX - adresa handlerului int h; Instalați un nou handler INT h Eliberați memorie din mediul DOS mov ah, h; ; Funcția DOS h mov es word ptr envseg ; ; ES - adresa segmentului de mediu DOS int h; Eliberați memoria Programe pentru rezidenți eu Lăsați rezidentul programului mov int initialize dx,offset initialize h endp; DX - adresa primului byte de la sfârşit; parte rezidentă ; Încheiați execuția cu ; rezident ems driver db 'EMMXXXHO', Numele driverului EMS de verificat ; Textul pe care programul îl afișează când pornește: utilisation db „Un program simplu pentru a copia ecranul numai de pe” db 'mod video h',ODh,OAh db „Alt-G scrieți copierea ecranului în scrgrb bmp” db Odh,Oah db 'scrgrb com /and - descărcare din memorie',ODh,OAh db '$' ; Textele pe care programul le emite la executarea cu succes: emsjnsg db „Încărcat în EMS”, ODh OAh, „$” convjnsg db „Neîncărcat în EMS”, ODh, OAh,”$” unloadedjnsg db 'Programul a fost descărcat cu succes din memorie',ODh,OAh, ; Texte pe care programul le emite în caz de erori: alreadyjnsg db 'Eroare: Programul deja încărcat',ODh,OAh,'$ ' no more mux msg db „Eroare: Prea multe programe rezidente” db Odh,Oah, cant unload jnsg db 'Eroare: Programul nu a fost găsit în memorie',ODh,OAh, '$' cant unload jnsg db „Eroare: Un alt program a interceptat întreruperi” db Odh,OAh,•$' descărcarea db •» dacă am fi început cu comutatorul /i ; Fișier BMP (pentru imaginea x x ) MP header labei byte ; Antetul fișierului BMP file header db "VM" Semnătura dd bfsize Dimensiunea fișierului dw , dd bfoffbits BMP data adresa de început ; Antet de informații BMP info header dd bl size Dimensiunea BMP info header dd Latime dd Înălțime dw Număr de planuri de culoare dw t Număr de biți pe pixel dd Metoda de comprimare a datelor dd * Dimensiunea datelor dd B h rezoluție X (pixeli pe metru) dd B h Rezoluție Y (pixeli pe metru) dd t Numărul de culori utilizate ( - toate) dd Număr de culori importante ( - toate) bi size = $-BMP info header Dimensiunea antetului BMP info header BMP header length = $-BMP header Lungimea ambelor anteturi bfoffbits = $-BMP file header+ * Dimensiunea antetului +' dimensiunea paletei Trucuri complexe de programare bfsize = $-BMP file header+ " + * ; Dimensiunea antetului + ; dimensiunea paletei + dimensiunea datelor lungimea programului = $-start sfârşitul începutului În acest exemplu, care este destul de complicat din cauza necesității de a evita toate cazurile de recuperare a întreruperilor DOS și BIOS, a fost adăugată o altă precauție - salvarea stării memoriei EMS înainte de a lucra cu aceasta și restabilirea acesteia la starea inițială Într-adevăr, dacă rezidentul nostru este activat în momentul în care un program funcționează cu EMS, și nu îndeplinește această cerință, programul nu va mai citi/scrie pe propriile pagini EMS, ci pe ale noastre Măsuri de precauție similare ar trebui luate ori de câte ori sunt apelate funcții care afectează orice structuri de date globale De exemplu: funcțiile de căutare de fișiere folosesc un buffer DTA, a cărui adresă trebuie salvată (funcția DOS Fh), apoi își creează propriul (funcția DOS IAh) și în final restaurează DTA procesului întrerupt la adresa salvată (funcția IAh) Astfel, este necesar să salvați / restabiliți starea liniei de adresă A (funcțiile XMS h și h), dacă programul rezident stochează o parte din datele sau codul său în zona NMA, salvați starea driverului mouse-ului (INT h , funcțiile h și h), salvați informații despre ultima eroare DOS (funcțiile DOS h și D Ah) și așa mai departe pentru fiecare resursă pe care o atinge TSR Scrierea de programe rezidente cu drepturi depline în DOS este foarte dificilă, dar atâta timp cât nu treceți dincolo de modul real, este cel mai eficient mijloc de a gestiona sistemul și de a face tot ce puteți face în DOS Programe semi-rezidente Programele semi-rezidente sunt programe care încarcă și execută un alt program rămânând în memorie, iar apoi, după ce programul încărcat se termină, se termină și ele normal Un program semi-rezident poate conține manipulatori de întreruperi care vor funcționa tot timpul în timp ce rulează un program obișnuit încărcat de sub el Deci, din punctul de vedere al acestui program copil, programul semirezident funcționează ca un program rezident obișnuit Aceste programe sunt utile pentru a face modificări și completări la programele existente dacă nu puteți face corecturi direct la codul lor executabil Așa sunt create încărcătoarele pentru jocuri care își stochează codul în formă criptată sau ambalată Un astfel de încărcător poate urmări anumite combinații de taste și poate înșela jocul adăugând anumite resurse la jucător sau, de exemplu, poate găsi un cod de verificare a parolei și îl poate opri De exemplu, să scriem un încărcător simplu pentru jocul Tee Fighter care va elimina introducerea parolei necesară de fiecare dată când jocul este lansat Desigur, acesta este un exemplu ipotetic, deoarece jocul nu își criptează fișierele în niciun fel, iar același efect ar putea fi obținut schimbând doar doi octeți în fișierul front ovl Singurul avantaj al programului nostru de descărcare va fi că este potrivit pentru toate versiunile jocului (de la X-Wing la Tie Fighter: Defender of the Empire) Programe pentru rezidenți III I LIII tieload asm Un exemplu de program semi-rezident este un program de descărcare care elimină verificarea parolei pentru jocurile Lucasarts: X-Wihg, X-Wing: Urmărire imperială, B-Wing, Tie Fighter, Tie Fighter: Apărătorul Imperiului modelul minuscul cod Pentru echipa LSS org h ; Programul COM start: ; Memoria liberă după terminarea programului (+ stivă) mov sp,lungimea programului ; muta stiva, mov ah, Ah ; Funcția DOS Ah mov bx,par length ; Dimensiunea în paragrafe int h; Modificați dimensiunea memoriei alocate ; Completați câmpurile EPB care conțin adrese de segment \ mov ax,cs • cuvânt mov ptrEPB+ ,ax mov cuvânt ptr EPB+ ,ax mov cuvânt ptr EPB+OCh,ax ; Încărcați programul fără execuție mov bx, offset EPB ; ES:BX-EPB mov dx, nume de fișier offset; DS:DX - numele fișierului (TIE EXE) mov ax, B h /funcția DOS B h int h; Încărcare fără execuție jne program loaded ; Dacă TIE EXE nu este găsit, mov byte ptr XWING ; setați flag pentru find passwd movax, B h mov dx, offset nume de fișier ; și încercați BWING EXE int h jne program loaded ; Dacă nu este găsit, movax, B h mov dx, offset nume de fișier ; încercați XWING EXE int h jc error exit ; Dacă nu este găsit (sau nu este încărcat ; din alt motiv) -; ieșiți cu un mesaj de eroare program loaded: ; Rutina de verificare a parolei nu este direct în executabil ; tie exe, bwing exe sau xwing exe, dar încărcate ulterior din partea frontală оѵi suprapunere, ; bfront ovl sau respectiv frontend ovl Găsiți comenzi care citesc; din această suprapunere și instalați handlerul nostru find passwd pe ele , cld apăsați CS pop topor adăugați ax, par length mov ds,ax MMIIII Trucuri complexe de programare xor si,si ; DS:SI este primul paragraf după încheierea programului nostru; (adică începutul zonei în care a fost încărcat programul modificabil) mov mov caii jc di,offset read file code ; ES:DI - cod pentru comparație, cx,rf code l ; CX este lungimea lui find string ; Căutare cod error exit ; Dacă nu este găsit, ieșiți din ; cu un mesaj de eroare ; Înlocuiți mov mov mov mov octeți din codul găsit cu caii find passwd și apoi comenzi byte ptr[si], Ah' ; Apelați (la distanță) word ptr [si+ ], offset find passwd word ptr [si+ ], cs byte ptr [si+ ], h ; NOP Rulați programul descărcat Este necesar să scrieți valorile inițiale corecte \u b\u în registrele pentru programul EXE ; și completați câteva câmpuri ale PSP-ului ei ; Funcția DOS h ; BX = segmentul PSP al programului încărcat; Pune-l în DS; și ES Completați și câmpurile PSP: exit without msg ; "adresa de retur"; și „adresa PSP a părintelui” ; Descărcați SS:SP; și transferă controlul către ; punct de intrare în program tip de protecție X-wing/Tie-fighter mov int mov mov mov mov mov« Iss jmp XWING ah, h h ds,bx es, bx word ptr ds:[OAh],offset word ptr ds:[ Ch],cs word ptr ds:[ h],cs sp,dword dword pt db ptr cs r cs:EP! EPB SSSP LCSIP ; / :T EPB dw ; Executabilul primește un mediu DOS din ; încărcătură în fagure, dw h,? ; și linia de comandă dw OO Ch,? ; și primul FCB, dw ooech? ; iar al doilea FCB EPB SSSPdd ? ; SS inițial:SP - completat de DOS EPB CSIP dd ? ; CS inițial: IP - completat de DOS filename db "tie exe", ; Mai întâi încercați să rulați acest fișier, filename db ' "bwing exe", ; apoi acesta, ■ filename db "xwing ehe", ; si apoi acesta ; Mesaje de eroare errorjnsg db „Eroare nu a găsit niciunul dintre fișierele TIE EXE,” db „BWING EXE, XWING EXE”, Odh, OAh, „$” error msg db „Fragment de cod de eroare nu a fost găsit”,ODh,OAh,'$' ; Comenzi care citesc fișierul de suprapunere în tie exe/bwing exe/xwing exe: read file code: db h, D h ; xor dx,dx db ■ B h, Fh ; movah, Fh db CDh, h ; int h j Programe pentru rezidenți db h ; jz ;(la adresa diferita in xwing si tie) rf code l = $-read file code ; Comenzi care apelează procedura de verificare a parolei ; Un set similar de comenzi se găsește în altă parte, deci find passwd ; va efectua verificări suplimentare passwd code: db h, h, FCh ; mov[bp- ],ax db h, h, FEh ; mov[bp- ],dx db h ; împinge dx db h; împinge toporul db Ah ; caii departe passwd l = $-passwd code astfel ieșire: mov jmp dx, offset errorjnsg short exit with msg ; Ieșire mesaj de eroare error exit : mov dx,offset error msg ; Ieșire mesaj de eroare exit with mag: mov ah, funcția DOS h: int h; imprimați șirul pe ecran exit without msg ■ ' • Controlul este de asemenea transferat aici după finalizare a programului încărcat (această adresă a fost introdus în câmpul PSP „adresă de retur”) mov ah, Ch ; Funcția DOS Ch: int h; sfârşitul programului ; Această rutină este apelată de fiecare dată de tie exe/bwing exe/xwing exe ; se citește din fișierul suprapus firid passwd proc departe ; Rulați cele trei comenzi pe care le-am înlocuit cu caii find passwd xorg dx dx mov ah, Fh ; Funcția DOS Fh: int h; citind dintr-un fișier sau dispozitiv punct dezactivare: ; La această adresă vom scrie codul de comandă RETF, , • când sarcina noastră este încheiată pushf Salvare steaguri push ds ; și registre împinge es pusha împinge cs pop es mov si dx ; DS:DX - începutul secțiunii tocmai citite fișier suprapus mov di,offset passwd code ; ES:DI - cod pentru comparație dec si Foarte curand o vom creste inapoi search for pwd: ; În acest ciclu, aparițiile găsite ale codului de referință ; sunt verificate pentru potrivirea exactă cu codul de verificare; parola | HH Trucuri complexe de programare inc si mov caii jc ; find string ; proceduri -cmp jne cmp jne , cmp jne jmp check for tie: cmp jne pwd found: mov mov mov cx,passwd l find string pwd not found a găsit altul, verificați sigur byte ptr [si+ ], h search for pwd byte ptr cs:XWING, check for tie cuvânt ptr [si+ ], h search for pwd scurt pwd found ; find string returnează DS:SI ; arătând spre începutul codului găsit – astfel încât ; uitați mai departe, trebuie să creșteți SI cu cel puțin ; Lungimea codului de referință ; Căutându-l în memorie ; Dacă nu este găsit, ieșiți introducerea codului nostru de apel de referință dacă este un apel la procedura de verificare a parolei ; Acest octet trebuie să fie ; În cazul aripii X/aripii B ; comanda je ar trebui să fie aici, mov pwd not found: popa POP pop popf ret find passwd Procedura Efectuează ES:OI- CX este DS:SI ei CF= CF= ; Intrare: Ieșire: in caz contrar: cuvânt ptr [si+ ], h search for pwd ; Deci procedura apel ; iar în cazul Tie Fighter -; Aici a fost găsită verificarea parolei - dezactivează-l word ptr ds:[sl+ ], h ; cuvânt ptr ds:[si+ ], h ; byte ptr ds:[si+ ], h ; ; și dezactivează-ne byte ptr cs:deactivation point,OCBh ; RETF NOP NOP NOP NOP NOP procedura find passwd ; Restaurați registrele es ds endp ; și steaguri ; și revine controlul la program find string căutați un șir de la o anumită adresă până la sfârșitul întregii adrese comune a șirului de referință lungime adresa de la care se începe căutarea dacă șirul nu este găsit, și DS:SI - adresa de la care a fost găsit , O memorie linia find string push push push proc aproape topor bx / dx ; Salvați registre do cmp: mov cmp loop: push push push dx, h di si cx ; Căutați în blocuri de h ( octeți) Programe pentru rezidenți III LI gere cmpsb ; Comparați DS:SI cu șir pop cx pop si POP di je găsit^cod ; Dacă se potrivește - - ieșiți cu CF = incsi ; În caz contrar, creșteți DS SI cu , dec dx ; contor de scădere în DX jne cmp loop ; și, dacă este diferit de zero, continuați Am depășit un alt bloc de kiloocteți sub si, h ; Scade SI cu h movax, ds inc ah; și crește DS cu mov ds,ax cmp ax, h ; Dacă ajungem jb do cmp ; segment, adrese I - pop-dx; restabilirea registrelor pop bx pop ah stc'; Setați cf= ret; si iesi Controlul este transmis aici dacă șirul este găsit cod găsit: pop dx pop bx pop axe cіc ret ; Restaurați registrele ; Set CF = ; si iesi find string endp sfârşitul programului: durata programului = $-start+ h+ h ; Lungimea programului în octeți par length = lungimea programului + OFh par length = par length/ ; Lungimea programului în paragrafe ■ sfârşitul începutului Comunicarea între procese Chiar dacă DOS este un sistem de operare cu o singură sarcină, mai multe procese pot rula în același timp Aceasta înseamnă că sistemul în sine nu oferă facilități speciale pentru execuția lor simultană, în afară de capacitatea de a lăsa programe rezidente în memorie Prin urmare, pentru a organiza memoria partajată pentru mai multe procese, trebuie încărcat un program rezident pasiv care va suporta funcțiile de alocare a unui bloc de memorie (returnarea unui identificator), determinarea adresei unui bloc (prin identificatorul său) și eliberarea unui bloc de memorie bloc - aproximativ în același mod în care funcționează driverele EMS sau XMS Pentru a implementa multitasking, va trebui să rulați un program rezident activ care va intercepta întreruperea IRQ și, pentru fiecare ciclu al temporizatorului sistemului, va prelua controlul pe rând de la fiecare dintre Trucuri complexe de programare procesează și transmite-l la următorul Aproape nimeni nu implementează multitasking cu drepturi depline în DOS, atunci când fiecare proces are propria memorie și nu poate accesa memoria altui proces - există un mod protejat pentru aceasta, dar există implementări destul de simple pentru o versiune ușoară de multitasking - comutarea firelor Un fir de execuție este un proces care utilizează același cod și date ca și restul acelorași procese din sistem, dar diferă de acestea prin conținutul stivei și al registrelor Apoi, programul rezident (dispecer) pentru fiecare întrerupere a temporizatorului va salva registrele firului de execuție întrerupt în structura sa, va citi registrele următorului fir în coadă și va returna controlul, iar structurile și stivele tuturor firelor vor fi stocate în unele zonă de memorie publică special alocată Programul specificat trebuie să accepte, de asemenea, apeluri multiple cu un fel de întrerupere software - crearea unui fir de execuție, ștergerea unui fir de execuție și, de exemplu, transferul controlului către următorul fir de execuție în timp ce firul de execuție curent este într-o stare de așteptare Această simplitate se transformă în complexitatea scrierii firelor în sine Deoarece toate au un cod comun, absolut tot ce se află în codul unui fir trebuie să fie reintre În plus, firele de execuție creează o mulțime de probleme de sincronizare, ceea ce duce la faptul că fie în codul tuturor firelor de execuție, fie în rezidentul principal, va trebui să implementați semafore, cozi, semnale, bariere și toate celelalte structuri care se găsesc în pachete reale pentru lucrul cu fire Să încercăm să creăm un prototip rudimentar de astfel de multitasking în DOS (doar două fire) și să vedem câte probleme trebuie să ne confruntăm scrsvr asm ; Un exemplu de sarcină simplă care implementează thread-ul multitasking în DOS ; Afișează doi șerpi pe ecran, mișcându-se aleatoriu, ; fiecare dintre ele este controlată de propriul fir fir ; Controlul transferului între THREADS nu funcționează într-o fereastră DOS (Windows ) modelul minuscul cod RNG utilizează registre pe de biți org lOOh „Programul COM start: mov ax, h Mod video h: int h > x x caii init threads ; Inițializați dispecerul nostru ; Din acest moment, până când shutdown threads este apelat, două fire sunt executate cu același fir ; același cod și date, dar cu registre și stive diferite ; (într-un sistem real, acesta ar fi un call to fork sau o funcție similară) mov bx, culoare (albastru) împinge bp mov bp, sp Împingeți toate variabilele locale în stivă, pentru a asigura reintrarea push' Adăugând la X la fiecare pas x inc equ word ptr [bp- ] Programe pentru rezidenți apăsați ; Adăugarea la Y la fiecare pas y ips equ word ptr [bp- ] împingere - ; Adresa relativă a capului buffer-ului line coords coords head equ word ptr [bp- ] apăsați ; Adresa relativă a cozii buffer-ului line coords coords tail equ word ptr [bp- ] subsp, * ; line coords - tampon circular de coordonate punct mov·di sp mov cx, m v ax, ; Completați-l cu coordonatele ( , ) împinge ds pop es rep stosw line coords equ word ptr [bp-( * )- ] împinge OAOOOh papi; ES - adresa memoriei video bucla principala: ; Ciclul principal caii display line ; Afișează starea actuală a șarpelui ; Schimbați direcția la întâmplare împinge bx mov ebx, ; Probabilitatea de schimbare a direcției este de / cai z random ; Obțineți un număr aleatoriu de la la mov ax,word ptr x inc mov bx cuvânt ptr y inc test dx,dx; Dacă acest număr este , jz rot right ; hai sa facem dreapta dec dx ; si daca - jnz exit rot ; stânga ; se întoarce neg ax ; la stânga de grade xchg ax,bx ; dY = -dX, dX = dY jmp scurt exit rot rot right: neg bx ; Corect de grade xchg exit rot: ax, bx ; dY = dX, dX = dY cuvânt mov ptr x inc,ax ; Scrieți noi valori de increment mov cuvânt ptr y inc,bx pop bx; Restabiliți culoarea în VX Mutați șarpele cu o poziție înainte mov di,word Ptr coords head ; DI - adresa șefului mov cx word Ptr line coords[di] ; CX este un șir mov dx word Ptr line coords[di+ ] ; DX - coloană adăugați cx word ptr y inc ; Adăugați incremente adăugați dx cuvânt Ptr x inc adaugă di, ; DI - următorul punct în tampon si di, ; Dacă DI > , DI = DI - mov word ptr coords head,di ; Acum capul este aici zoo ] j I j I П Trucuri complexe de programare mov word ptr ,line coords[di],cx î' ; Notează-i coordonatele mov word ptr line coords[di+ ],dx mov di,word ptr coords tail Citiți adresa de coadă adaugă di, ; Mută-l unul si di, ; poziție înainte mov word ptr coords tail,di ; și scrie în loc DIN Pauză ■ Datorită particularităților dispecerului nostru (vezi mai jos), nu putem folosi întreruperea BIOS pentru a întrerupe, așa că vom face doar o buclă goală Lungimea ciclului va trebui modificată în funcție de viteza procesorului mov cx, - bucla$; comenzi în buclă mov cx, - bucla $ mov cx,- bucla $ muta ah, int ore; Dacă niciuna dintre taste nu a fost apăsată, iz main loop ; continuați bucla principală mov ah, ; În caz contrar, citiți cheia int părăsi; Stivă gratuită din variabile locale caii shutdown threads ; Dezactivați multitasking în acest moment avem din nou unul singur proces mov ax, ; Modul video : int h; x int h; Sfârșitul programului ; Procedura de afișare a unui punct pe ecran în modul h CX = rând, OX = coloană, BL = culoare, ES = OAOOOh putpixel proc aproape împinge di lea ecx,[ecx* +ecx] ; CX = rând x shl cx, ; CX = rând x x = rând x adăugați dx,cx; DX = rând x + coloană = adresă, mov di dx mov al, bl stosb ; Scrieți un octet în memoria video pop di ret putpixel endp ; procedura display line ; Afișează șarpele nostru pe ecran la coordonatele din memoria tampon line coords display line proc aproape mov di,word ptr coords tail ; Porniți ieșirea din coadă, continue line display: cmp di,word ptr coordsjnead ; Dacă DI este egal cu adresa principală, je line displayed ; ieșirea s-a terminat Programe pentru rezidenți eu caii adăuga și jmp line displayed: punct afișare di, di, short continue line display ; În caz contrar ^ afișează punctul pe ecran ; Setați OG : la următorul punct Si asa mai departe caii display point mov di,word ptr coords tail ; Desenați un punct la coadă împinge bx mov bx, ; culoare zero caii display point ; adică șterge POP cutie ret display line • endp ; procedura display point ; Afișează un punct display point mov mov caii ret display point din buffer line coords cu index DI proc aproape cx word ptr line coords[di] ; dx word ptr line coords[di+ ] ; putpixel ; Linia Coloană Ieșire punct endp ; z procedura aleatorie ; Generator standard de numere aleatoare congruente; Intrare: EBX - numărul maxim ; Ieșire: EDX - un număr de la la EBX- z random: (neoptimizat) zr cont: zr init: împinge ebx cmp byte ptr zr init flag, ; Dacă nu ai fost sunat încă je zr init ; fi inițializat mov eax,zr prev rand ; În caz contrar, înmulțiți precedentul mul rnd number ; multiplicator div rnd number ; și împărțiți cu divizor mov zr prev rand, edx ; Restul diviziunii este noul număr pop ebx mov eax,edx xor edx,edx div ebx ; Împărțiți-l la maxim' ret; și returnați restul la EDX push h Inițializare generator pop fs; h: Ch - mov eax,fs:[ Ch] ; Contor de întreruperi ale temporizatorului BIOS, mov zr prev rand,eax ; va fi primul număr aleatoriu mov byte ptr zr init flag, jmp zr cont ; Factor ; Divizor dd dd rnd number rnd number i LLLJ Trucuri complexe de programare zr init flag db ; Steagul de inițializare a generatorului zr prev rand dd O ; Numărul aleatoriu anterior ; Aici începe codul de dispecer pentru multitasking ; Structura de date în care stocăm registrele pentru fiecare fir thread struc struc ah dw? bx dw ? cx dw ? dx 'dw ? -Si dw? didw ? bp dw ? -SPdw? -IP dw? steaguri dw ? thread struct se termină procedura init threads ; Inițializează manipulatorul de întreruperi OBh și completează structuri care descriu ; ambele fire init threads proc aproape pushf pusha împinge es mov ax, h AH = h, AL - număr de întrerupere int h Definiți adresa handlerului mov word ptr old int h,bx Salvați-l mov word ptr old int h+ ,es mov ax, h AH = h, AL = numărul de întreruperi mov dx,offset int h handler Setați-l pe al nostru int h POP es popa ■; Acum registrele sunt aceleași ca la apelarea procedurii popf mov thread ax,ax Umpleți structuri mov thread ax,ax threadl și thread , mov thread bx, bx unde este stocat conținutul mov thread bx,bx din toate registrele (cu excepția celor de segment) mov threadl cx,cx nu se schimba in acest exemplu mov thread cx,cx mov threadl dx,dx fir mov dx,dx mov threadl si, si mov thread si,si mov threadl di,di mov thread di,di mov threadl bp,bp fir mov bp,bp; ț Programe pentru rezidenți eu mov thread sp, offset th'read stack+ mov thread sp, offset thread stack+ topor pop ; Adresă de retur (stiva este acum goală) fir mov ip,ax mov 'thread ip,ax pushf topor pop ; Steaguri fir mov steaguri,ax fir mov steaguri,ax ' mov sp,thread sp ; Setați stiva de fire jmp cuvânt ptr thread ip ; si da-i controlul init threads endp curent fir db • ; Numărul curent al firului ; Manager de întreruperi INT h (IRQO) ; schimbă firele intO h handler proc departe pushf ; Sunați mai întâi vechiul administrator db Ah ; Cod de comandă caii departe old int h dd ; Adresa vechiului handler ; Determinați dacă a avut loc o întrerupere în timp ce firul nostru de discuții se executa sau ; alt handler de întrerupere Acest lucru este important deoarece nu vom face ; returnează controlul programului altcuiva, cel puțin pentru moment ; Acesta este motivul pentru care nu putem folosi întreruperi pentru întârzieri în firele noastre ; iar programul nu rulează într-o cutie DOS (Windows ) mov save dl, bp ; Salvați BP mov bp sp împinge toporul împinge bx pushf mov ax,word ptr [bp+ ] ; Citiți partea de segment mov bx, cs ; adresa expeditorului cmp ax bx ; Comparați-l cu CS jne numit far ; Dacă nu se potrivesc, ieși popf pop bx; În caz contrar - restaurați registrele pop topor mov ѣр, save di mov save di,di ; Salvați DI, SI mov save si,si pushf , ; și steaguri ; Stabiliți din ce fir la care este necesar să transferați controlul str byte ptr curent fir, ; Dacă din prima Je thread to thread ; accesați thread to thread mov byte ptr curent fir, ; Dacă de la la , scrieți la număr mov si,offset threadl ; și instalați SI și DI; mov di,offset thread ; la structurile corespunzătoare jmp short order selected ^ ygtp Trucuri complexe de programare thread to thread : ; Dacă de la la , mov byte ptr current thread, ; notează fire numărul mov si,offset thread ; și instalați SI și DI, mov di, offset threadl order selected: ; Scrieți toate registrele curente în structura la adresa [DI] ; și încărcați toate registrele din structură la adresa [SI], ; începe cu SI și DI: mov ax,[si] si ; Pentru MASM, toate expresiile [reg] reg trebuie să fie push save si ; înlocuiți (thread struct Ptr[reg]) reg pop[di] si mov save si,ax mov ax, [si] di push save di pop[di] di mov save di,ax Acum toate registrele principale mov[di ax],ax mov ax,[si, ax] mov rt[di bx],bx mov bx,[si bx] mov[di cx],cx mov cx,[si cx] mov[di dx],dx mov dx,[si dx] mov[di bp],bp movbp,[si bp] Steaguri ■ pop[di flags] împinge [si flags] popf adresa expeditorului pop [di ip] ; Adresa de retur din stivă se adaugă sp, ; CS și steaguri din stivă - acum este gol ; Schimbați stivele mov [di sp], sp mov sp,[si sp] push [si ip] ; Adresă de întoarcere la edem (deja nouă) mov di,save di ; Sarcina și SI mov si, s'ave si retn; și săriți la adresa din stivă ; Controlul trece aici dacă întreruperea a avut loc în codul altcuiva numit departe: popf L; Restaurați registrele POP cutie "Toporul pop"- mov bp,save di iret; și terminați handler-ul int h handler endp save di dw ? ; Variabile pentru stocarea temporară save^si 'dw ? ; registre ; Procedura shutdown threads ; Oprește controlerul shutdown threads proc peg movax, h ; Este suficient doar să restabiliți întreruperea Id-uri dx,dword ptr old int h int h ret shutdown threads endp ; O structură care descrie primul fir threadl thread struc despre ; Iar al doilea thread thread struc despre ; Teancul primului fir thread stack db dup(?) ; Iar al doilea thread sack db dup(?) r \ sfârşitul începutului După cum putem vedea, acest exemplu nu poate funcționa pe Windows și în alte cazuri în care DOS este extins la un sistem de operare mai avansat De fapt, în acest exemplu, am făcut exact asta - am implementat un fragment din sistemul de operare care nu este în DOS Într-adevăr, folosind mecanismul de gestionare a întreruperilor, este posibil să creați un sistem de operare în mod real similar cu DOS, dar se va dovedi foarte repede că pentru aceasta trebuie să comunicați direct cu hardware-ul computerului, adică să utilizați I / O porturi Programare la nivelul porturilor de intrare-ieşire După cum sa văzut în capitolul anterior, utilizarea funcțiilor sistemului DOS și a întreruperilor BIOS poate fi nesigură din cauza lipsei de reintrare Acum este momentul să cobori la următorul nivel și să înveți cum să lucrezi direct cu dispozitivele computerului, prin porturile I/O, așa cum fac funcțiile sistemului În plus, multe caracteristici ale computerului pot fi realizate doar prin programare la nivel de port Tastatură Controlerul tastaturii are porturi numerotate de la h la Fh, deși porturile h și h sunt suficiente pentru toate operațiunile standard h pentru citire - registrul de stare a tastaturii, returnează următorul octet: bit : eroare de paritate la transmiterea datelor de la tastatură bit : timeout de primire bit : timeout de transmisie guşă Trucuri complexe de programare bit : tastatura închisă bit : date scrise în registrul de intrare - comandă bit : autotest finalizat bit : există date în tamponul de intrare (pentru un controler cu tastatură) bit : există date în tamponul de ieșire (pentru computer) Când este scris pe acest port, joacă rolul unui registru suplimentar de control al tastaturii, dar comenzile sale variază foarte mult pentru diferite plăci și diferite BIOS-uri și nu îl vom lua în considerare în detaliu h pentru citire și scriere - registru de control al tastaturii Dacă bitul înalt al acestui port este scris la , tastatura va fi blocată, dacă este deblocat Alți biți din acest port nu pot fi modificați, deoarece controlează alte dispozitive (în special, difuzorul) Pentru a schimba starea tastaturii, trebuie să citiți un octet din port, să schimbați bitul și să scrieți din nou acest octet în portul h h pentru citire - portul de date de la tastatură Când citiți din acesta, puteți obține codul de scanare al ultimei taste apăsate (vezi Anexa ) - așa este cel mai bine să implementați programe rezidente care interceptează întreruperea IRQÎ, deoarece acest cod poate fi folosit pentru a determina momentul apăsării și eliberarea oricărei taste, inclusiv astfel de taste, cum ar fi Shift, Ctrl, Alt sau chiar Pauză (codul de scanare pentru eliberarea tastei este egal cu apăsarea codului de scanare plus h): int h handler: în al, h cmp al,hot key jne nu cheia noastră [ ] nu cheia noastră: jmp old int h ; Citiți codul de scanare al cheii ; Dacă aceasta este cheia noastră rapidă, mergeți la managerul nostru ; Acțiunile noastre sunt aici ; Îl sun pe vechiul handler Încă nu putem termina handler-ul doar cu o comandă IRET, pentru că mai întâi, handlerul de întrerupere hardware de la tastatură trebuie să seteze bitul al portului h și apoi să îl resetați, astfel: un al, h împinge toporul sau al, h afară h,al pop topor afară h,al În al doilea rând, trebuie să informeze controlerul de întrerupere că procesarea întreruperii hardware s-a încheiat cu comenzile mov al, h afară h,al h pentru scriere - registru de control al tastaturii Un octet scris pe acest port (dacă bitul din portul h este ) este interpretat ca o comandă niste Programare la nivel de port comenzile constau din mai mult de un octet - atunci ar trebui să așteptați ca acest bit să fie șters din nou înainte de a trimite următorul octet Enumerăm cele mai comune comenzi Comanda OEDh ?h - schimba starea diodelor tastaturii Al doilea octet al acestei comenzi definește noua stare: bit : stare de blocare derulare ( - activat, - dezactivat) bit : stare Num Lock bit : stare Caps Lock Acest lucru nu schimbă starea comutatoarelor stocate de BIOS în octeții de stare a tastaturii, iar gestionarea de întrerupere a tastaturii din BIOS va restabili starea LED-urilor cu prima ocazie Comanda OEEh este o cerere de ecou Tastatura răspunde cu un cod de scanare OEEh Comanda F h ??h - setați parametrii modului de repetare automată: bitul al celui de-al doilea octet de comandă: biții - : pauză înainte de începerea repetării automate: b - ms, b - ms, b - ms, b = lOOOms biți - : setați rata de repetare automată (caractere pe secundă): OOOOO- b- b - b = b- b- b- b- b- , b- , Toate valorile intermediare au, de asemenea, sens și corespund vitezelor intermediare, de exemplu b - Comanda F h - activați tastatura Comanda F h - opriți tastatura Comanda F h - setați parametrii impliciti Comanda OFEh - trimiteți din nou ultimul cod de scanare Comandă OFFh - efectuați autotestarea Tastatura răspunde la toate comenzile, cu excepția OEEh și OFEh, cu un cod de scanare OFAh (confirmare), care este absorbit de handlerul BIOS standard, așa că dacă nu îl înlocuim complet, nu trebuie să ne facem griji cu privire la procesarea OFAh Ca exemplu de lucru cu o tastatură, să ne uităm direct la un program simplu care comută LED-urile mig asm Parcurge LED-urile tastaturii org model minuscul cod h Programul COM i Trucuri complexe de programare începe proc aproape mov ah, ; Funcția Întreruperea IAh: Int IAh ; obține ora curentă mov ch,dh ; Stocați secunda curentă în CH mov cl, b ; CL = starea LED-ului tastaturii bucla principala: caii change LEDs ; Instalați LED-uri conform CL shl cl, ; Următorul LED test cl, b • ; Dacă a ieșit un în bitul , jz continua mov cl, b ; întoarce-l la bitul continua: mov; ah, ; Verificați dacă o tastă a fost apăsată int h jnz exit loop ; Dacă da, părăsiți programul împinge cx mov ah, ; Funcția întrerupe Ah int Ah ; Obțineți ora curentă pop cx cmp ch, dh ; Comparați secunda curentă în DH cu CH mov" ch,dh ; Copiați-l oricum continui; Dacă a fost aceeași secundă -; nu comuta LED-urile jmp short main loop ; În caz contrar, comutați LED-urile exit loop: mov ah, ; Ieșire în buclă - a fost apăsată o tastă int h Citiți ret; și încheiați programul start endp ; procedura de schimbare LED-uri ; Setează starea LED-urilor tastaturii în funcție de numărul din CL Change LEDs proc aproape caii wait KBin ; Se așteaptă trimiterea unei comenzi mov al OEDh afară h,al ; Comandă de la tastatură EDh caii wait KBin ; Se așteaptă trimiterea unei comenzi mov al, cl afară h,al ; Stare nouă a LED-urilor ret change LEOs endp procedura wait KBin Se așteaptă introducerea unei comenzi de la tastatură wait KBin p roc nea r în al, h; Citiți cuvântul de stare test al, b; Este bitul egal cu ? jnz wait KBin ; Dacă nu, așteptați ret; Dacă da, ieși wait KBin endp ■ sfârşitul începutului Programare la nivel de port Port serial Fiecare dintre porturile seriale comunica cu procesorul printr-un set de porturi I/O: COM - F h - FFh, COM - F h - FFh, COMZ - E h - EFh si COM - E h - EFh Numele de porturi COM -COM nu sunt de fapt fixate în niciun fel BIOS-ul denumește pur și simplu portul COM , a cărui adresă ( F h implicit) este stocată în zona de date BIOS la h: h În mod similar, portul COM , a cărui adresă este scrisă la h: h, COM este h: h și COM este h: h Luați în considerare alocarea porturilor I/O folosind exemplul F h - FFh F h pentru citire și scriere - dacă bitul înalt al registrului de control al liniei = , atunci acesta este registrul de transfer de date (THR sau RBR) Transmiterea și recepția datelor prin portul serial corespund scrierii și citirii pe acest port particular F h pentru citire și scriere - dacă bitul înalt al registrului de control al liniei == , atunci acesta este octetul mic al divizorului de porturi F h pentru citire și scriere - dacă bitul înalt al registrului de control al liniei = , atunci acesta este registrul de activare a întreruperii (IER): bit : întrerupere de schimbare a stării modemului bit : BREAK sau întrerupere de eroare bit : anulați dacă bufferul de transmisie este gol bit : întrerupeți dacă au sosit date noi F h pentru citire și scriere - dacă bitul înalt al registrului de control al liniei = , atunci acesta este octetul înalt al divizorului de porturi Valoarea vitezei portului este determinată de valoarea divizorului de frecvență (vezi Tabelul ) FAh pentru citire - registru de identificare întrerupt Conține informații despre motivul întreruperii pentru handler: biții - : - fără FIFO, - FIFO prezent bit : Timeout FIFO receptor biții - : tipul de întrerupere care a avut loc: b - stare BREAK sau eroare Resetați după citirea de la FDh b - au venit datele Resetați după citirea de la F h b - bufferul de trimitere este gol Sterge după scrierea la F h b - starea modemului s-a schimbat Resetați după citirea de la FEh bit : dacă a avut loc o întrerupere; dacă nu există FAh de scris - registru de control FIFO (FCR) biții - : pragul de întrerupere a recepției: b - octet b - octeți b - octeți b - octeți wJ LJ li Trucuri complexe de programare bit : clear transmitter FIFO ; bit : Ștergeți receptorul FIFO bit : activați funcționarea FIFO FBh pentru citire și scriere - registru de control al liniilor (LCR) bit : dacă - porturile F h și F h funcționează ca un divizor de port bitul : stare BREAK - portul trimite continuu zerouri biții - : paritate: ? ? - fără paritate - paritate impară - paritate - paritate fixă - paritate fixă ? ? - bitul de paritate software (nu hardware) : numărul de biți de oprire: - bit de oprire I- biți de oprire pentru -, -, -biți; , biți de oprire pentru cuvinte de biți biți - : lungimea cuvântului: - biți - biți - biți II- biți FCH pentru citire și scriere - registru de control al modemului (MCR) bit : diagnosticare (ieșirea portului COM scurtat la intrare) bit : linie OUT - trebuie să fie pentru ca întreruperile să funcționeze bit : linie OUT - trebuie să fie bit : RTS linia bit : linia DTR FDh Citire - Registrul de stare a liniei (LSR) bit : registrul de deplasare emițător gol bit : registrul de reținere al transmițătorului este gol - poate fi scris în F h bit : starea BREAK detectată (șir de zerouri mai lung decât bitul de pornire + cuvântul + paritatea + bitul de oprire) bit : eroare de sincronizare (zero bit stop primit) bit : eroare de paritate bit : eroare de overflow (a sosit un nou octet, deși cel vechi nu a fost citit de la F h, în timp ce vechiul octet este pierdut) bit : date primite și gata pentru a fi citite de la F h FEh pentru a citi - registrul de stare modem (MSR) bit : linie DCD (purtător) bit : linie RI (apel) bit : linie DSR (date gata) bit : linie CTS (permisiunea de a trimite) bit : starea DCD a fost schimbată bit : starea RI a fost schimbată Programare la nivel de port II bit : starea DSR a fost schimbată Q bit: starea CTS a fost schimbată FFh pentru citire și scriere - registru de rezervă Neutilizat de controlerul portului serial, orice program îl poate folosi Tabelul Divizoare porturi seriale Divizor de frecvență Viteza h ' h h h OOOCh b b b h Deci, primul lucru pe care trebuie să-l facă un program care lucrează cu un port serial este să-l inițializeze prin scrierea numărului h în registrul de control al liniei ( FBh), divizorul de frecvență în porturile F h și F h, modul în portul FBh și, de asemenea, specificând întrerupere activată în portul F h Dacă programul nu folosește deloc întreruperi, scrieți pe acest port Înainte de a scrie date pe portul serial, puteți verifica bitul , iar înainte de a citi - bitul al registrului de stare a liniei, dar dacă programul folosește întreruperi, aceste condiții sunt îndeplinite automat În general, munca serioasă cu portul serial este posibilă numai cu ajutorul întreruperilor Să vedem cum poate fi aranjat un astfel de program folosind următorul exemplu: ; termen asm ; Program terminal minim care utilizează întreruperi ; Ieșire - Alt-X modelul minuscul cod org h ; Programul COM ; Următoarele patru directive determină ce port serial ; programul a fost compilat (nu se efectuează verificări - nu rulați acest lucru ; exemplu dacă nu aveți un modem pe portul corespunzător) Program real ; trebuie să specifice numărul portului din fișierul de configurare sau din linia de comandă COM equ F h ; Numărul portului de bază (C M ) IRQ equ OBh ; Număr de întrerupere (INT JVI pentru IRQ ) E BITMASK equ b Bitmask pentru a activa IRQ D BITMASK equ b ; Bitmask pentru a dezactiva IRQ III IUI Trucuri complexe de programare start: caii, ihit everything ; Inițializare linie și modem bucla principala: ; Ciclul principal ; Programul terminal real din această buclă va scoate date din buffer; recepție (completată de la handler-ul de întrerupere) pe ecran, dacă există un normal; lucru, la un fișier, dacă un fișier este trimis sau procesat într-un fel diferit ; În exemplul nostru, folosim bucla principală pentru a introduce caractere, deși acest lucru este mai bun; face de la manipulatorul de întreruperi de la tastatură mov ah, ; Funcția DOS h: int h; citind cu așteptare și fără ecou test al,al ; Dacă este introdus un caracter obișnuit, jnz send char ; Trimite-l int h; În caz contrar, citiți codul ASCII extins cmp al, Dh ; Dacă nu este Alt-X, jne main loop ; continua ciclul caii shutdown everything ; În caz contrar, restaurați totul la ; starea initiala ret; și încheie programul trimite car: ; Trimiterea unui caracter la modem ; Un program de terminal real ar trebui să adauge doar un caracter la buffer aici ; transfer și, dacă acest buffer era gol, activați întreruperile „registru de transfer ; gol" Doar trimite personajul direct la port movdx com ; Registrul THR afară dx,al jmp scurtă buclă principală old irqdd ? ; Aceasta va stoca adresa vechiului ;handler Handler simplificat de întrerupere a portului serial irq handler proc departe pusha; Salvați registre mov dx,C M+ ; Citiți registrul de identificare în al, dx; întrerupe repeatjiandler: și ax, b ; Setați toți biții la zero, cu excepția și mov , di , ax ; responsabil pentru situații principale caii word ptr cs:handlers[di] ; Apel de procedură indirectă; pentru a gestiona situația mov dx,C M+ ; Citiți din nou registrul de identificare în al, dx; întrerupe test al, ; Dacă bitul cel mai puțin semnificativ nu este , jz repetare handler; trebuie tratată o altă întrerupere mov al, h; În caz contrar, opriți întreruperea hardware afară h,al ; prin trimiterea comenzii E I (vezi secțiunea ) popa iret ; Tabel de adrese al procedurilor care deservesc diferite opțiuni de întrerupere, handler-uri dw offset line h, offset trans h dw offset recv h, offset modem h Programare la nivel de port ; Această procedură este apelată atunci când starea liniei se modifică line h y proc peg mov dx,C M+ ; Până când LSR-ul este citit, în al dx; întreruperea este considerată incompletă ; Aici puteți verifica ce s-a întâmplat și, de exemplu, puteți întrerupe conexiunea dacă ; A fost detectată starea BREAK ret line h endp ; Această procedură este apelată atunci când sunt primite date noi recv h proc aproape movdx com ; Până la citirea RBR, în al,dx ; întreruperea este considerată incompletă ; Aici, octetul primit ar trebui să fie plasat în tamponul de primire pentru programul principal, ; dar o vom afișa imediat int h,; Ieșire pe afișaj 'ret recv h endp ; Această procedură este apelată când transferul de date este finalizat țransjh proc lângă ; Aici, următorul caracter din buffer-ul de transmisie ar trebui să fie scris în THR și, dacă ; tamponul este apoi gol, dezactivați acest tip de întrerupere ret trans h endp ; Această rutină este apelată atunci când starea modemului se schimbă modem h proc aproape mov dx,C M+& ; Până la citirea MCR, în al dx; întreruperea este considerată incompletă ; Aici puteți determina starea apelului și puteți ridica receptorul, determina ; pierderea operatorului și apelul înapoi etc ret modem h endp irq handler endp ; Inițializarea a tot ceea ce trebuie inițializat init everything proc aproape Se instalează handlerul nostru de întreruperi movax, h+IRQ; AH = h, AL = numărul de întreruperi int h; Obțineți adresa vechiului handler mov word ptr old irq,bx ; și salvați în old irq mov word ptr old irq+ ,es movax, h+IRQ; AH = h, AL = numărul de întreruperi mov dx offset irq handler ; DS:DX este responsabilul nostru int h; Instalați un nou handler ; Resetați toate registrele de porturi movdx,COM+ ; Registrul IER mov al, out dx,al '] ; Dezactivați toate întreruperile mov dx,C M+ ; MCR out dx,al ; Resetați toate liniile de modem la mov dx,C M+ ; și citit din LSR, Trucuri complexe de programare în al, dx movdx,COM+O ; de la RBR în al, dx movdx,COM+ ; și de la MSR în al, dx; în cazul în care s-au schimbat recent mov dx,C M+ ; și trimite, de asemenea, la registrul FCR, mov al, ; pentru a opri FIFO afară dx,al Setați viteza portului COM mov dx,C M+ ; Scrieți la registrul LOR mov al, h; orice număr cu bitul cel mai semnificativ out•dx al movdx,COM+O ; Acum scrieți în registrul DLL mov al, ; octet mic al divizorului de viteză, afară dx, al movdx,COM+ ; și în DLH - mov al, ; octet mare out dx, al ; (am notat I - viteza portului ) Inițializarea liniei movdx,COM+ ; Scrie acum la LCR mov al, b; număr corespunzător modului N out dx,al ; (cel mai des folosit) Inițializarea modemului mov dx,C M+ ; Scrieți la registrul MCR mov al, b; masca de biți care permite DTR, RTS out dx, al ; și UT Aici ar trebui să verificați prezența unui modem pe acest port (citiți registrul MSR până când liniile CTS și DSR sunt setate sau expiră timpul), apoi trimiteți la modem (adică puneți în buffer-ul de transmisie) un șir de inițializare, de exemplu „ATZ”, O Dh Activați întreruperile movdx,COM+ ; Scrieți la IER o mască de bit care permite mov al, b; toate întreruperile cu excepția „registrului de transfer gol” afară dx,al în al, h; Citiți CW (vezi secțiunea ) și al E BITMASK ; Demascați întreruperea afară h,al ; Scrieți CW , ja exitjnodex ; procedura de iesire ' ; (ramanand in modul h) shl di, ; Înmulțiți cu deoarece xjnodes -; tabel de cuvinte mov di,word ptr x modes[di] ; Citit ; adresa tabelului de setări pentru ; modul selectat mov dx, C h ; Port C h - index sincronizator movax, ODh ; Înregistrați OOh, valoarea out dx,ax ; Resetare asincronă mov ax, h ; Înregistrare h, valoare h out dx,ax Dezactivează modul CHAIN mov dl, C h ; Port C h - Registrul MOR pentru scriere mov al,byte ptr [di] ; Scrieți-i valoarea ratei cadrelor ouț dx,al ; și polaritatea de baleiaj pentru modul selectat mov dl, D h ; Port D h - index controlor C RT mov si,word ptr offset [di+ ] Adresa șirului cu setări pentru lățimea selectată în DS:SI ' mov cx, ; Lungimea șirului de setări în CX rep outsw ; Ieșiți un șir de cuvinte în porturile / D mov si,word ptr offset [di+ ]; Setări pentru înălțimea selectată în DS:SI mov rep cx, ; outsw Lungimea șirului de setări în CX Programare la nivel de port mov si,word ptr offset [di+ ]; Setări / ;■ pentru a activa/dezactiva dublarea ; fără verticală (linii / și / ) mov cx, rep outw mov ax, word ptr offset [dl+ ] ; Numărul de octeți din șir mov word ptr x width,ax ; Stocați în variabila x width mov dl, C h ; Port C h - index sincronizator mov ax, b ; Înregistrare h, valoare ouț dx ax ; Ieșiți din starea de resetare exitjnodex: ret ; Tabel de adrese ale tabelelor cu setări de mod x modes dw offset mode ,offset mode dw offset mode , offset mode dw offset mode , offset mode dw offset mode , offset mode HOR, adresa de linie ; Tabel de setare a modului: valoarea înregistrată ; setări de lățime, setări de înălțime adresa de linie, adresa de linie ; setări de dublare verticală, număr de octeți pe linie mode mode mode mode mode mode mode mode dw dw dw dw dw dw dw dw h, offset h, offset h, offset h, offset E h, offset E h, offset OE h, offset E h, offset mode w,mod offset w,mod offset w,mod offset w,offset mode i w, offset mode w, offset mode w, offset mode w, offset mode h,mod offset double, / mode h,offset mode single, / mode h,offset mode double, / mode h,offset mode single, / mode h,offset mode h, offset mode h,mod offset mode double, / mode single, / mode double, / mode single, / CRT În fiecare cuvânt, octetul mic este valoarea care se află în acest registru; Setare - număr de registru, introdus, lățime ; Setări ; mai vechi -mode w: ; Primul registru este neapărat h, deși nu se referă la lățime, ; dar permite dw mode w: dw mod h: mod h: dw mod h: mod h: dw mode single: dw mode double: dw setmode x scrierea la alte registre, dacă a fost dezactivată (I) E h, F h, F h, h, b, O h, h, h ; Setare la de lățime E h, B h, h, A h, E h, E h, A h, D h ; Reglare pe inaltime / BF h, F h, C h,OETIh, F h, h, B h ; Reglare pe inaltime / D h, E h, EA h; C h, DF h, E h, b ; Setarea modurilor fără dublare h, h, E h ; Setarea modului de dublare h, h, E h endp Trucuri complexe de programare x width dw ? ; Numărul de octeți din șir ; Această variabilă este inițializată de setmode x și utilizată de putpixel x ; procedura putpixel x /Afișează un punct cu culoarea specificată în modul X curent ; Intrare: DX = șir ; CX = coloană ; BP = culoare ; ЁВ = OAOOOh ; DS = segmentul în care se află x width putpixel x pusha proc aproape movax; dx mul word ptr x width ; AX = patruzeci x număr de octeți pe linie pyv di,cx ; DI = coloană shr di, ; = coloană/ (număr de octeți pe rând) adauga di,ax ; = numărul de octeți din memoria video movax, h ; AL = h (număr de înregistrare) AH = (mască de biți) și cl, h; CL = rest după împărțirea coloanei la = numărul planului de culoare shl ah, cl ; Acum AN este setat la bit corespunzător planului de culoare dorit mov dx, C h ; Port S P - index sincronizator out dx,ax ; Permite scrierea doar în planul dorit movax,bp ; Culoare în AL stosb Trimite un octet în memoria video popa ret- putpixel xendp Registre VGA DAC ( C h - C h) Tabelul de culori VGA este de fapt de registre, fiecare dintre acestea conținând trei numere de biți corespunzătoare nivelurilor de roșu, verde și albastru Subfuncțiile INT Oh AX = Oh - Bh fac ca lucrul cu aceste registre să fie convenabil, dar dacă este necesară viteza maximă, programarea lor la nivel de porturi I/O nu este mult mai dificilă C h citire/scriere- Registrul de mascare a pixelilor (implicit OFFh) La accesarea registrului DAC, se efectuează o operație AND asupra numărului acestuia și a conținutului acestui registru C h pentru scriere, registru index DAC pentru modul citire Scrierea unui octet aici pune DAC-ul în modul de citire, astfel încât următoarea citire de la C h va returna valoarea registrului paletei la acel index Programare la nivel de port II I- C h Citire: Registrul de stare DAC biții - : b/ Ib - DAC în modul scriere/citire C h Citire/Scriere: registru index DAC pentru modul de scriere Scrierea unui octet aici pune DAC-ul în modul de scriere, astfel încât o ieșire ulterioară către C h va face ca noi valori să fie scrise în registrele paletei începând cu acel index C h citire/scriere: registru de date DAC Citirea de aici citește valoarea registrului paletei cu un index plasat anterior la C h, scrierea scrie o nouă valoare în registrul paletei cu un index situat la C h Sunt necesare trei citiri/scriere pentru fiecare registru, transferând trei valori ale nivelului de culoare pe biți: roșu, verde, albastru După a treia operație de citire/scriere, indexul registrului actual al paletei este incrementat cu , astfel încât mai multe registre pot fi citite/scrise simultan Comenzile insb / outsb facilitează foarte mult lucrul cu registrele DAC în cazurile în care este necesară citirea sau încărcarea secțiunilor semnificative ale paletei sau a întregii palete - astfel de proceduri sunt atât mai rapide, cât și mai mici decât cele similare scrise folosind întreruperea INT h Să vedem cum este implementat acest lucru folosind exemplul unui program de golire fluidă a ecranului fadeout asm Efectuează o acoperire ușoară a ecranului modelul minuscul cod ; Pentru comenzi insb/outsb org h ; Programul COM start: cld ; Pentru comenzile de procesare a șirurilor mov di, palete offset caii read palette ; Salvați paleta curentă astfel încât ; restaurare la sfârșitul programului, mov di, palete offset+ * caii read palette ; și, de asemenea, scrie o altă copie ; paleta curentă, care va fi ; modifica z mov cx, ; Contor de cicluri de schimbare a paletei bucla principala: împinge cx caii wait retrace Așteptați ca raza să înceapă să retrace mov di, palete offset+ * mov si,di caii dec palette ; Reduceți luminozitatea tuturor culorilor caii wait retrace ; Așteptați următorul mov si,offset palettes+ * ; fascicul invers caii write-palette ; Înregistrați o nouă paletă pop cx Trucuri complexe de programare buclă bucla principală ; Bucla este executată de de ori - suficient; pentru a elimina cea mai strălucitoare culoare ; (valoarea maximă de biți ; componente - ) mov caii ret si offset palete write palette ; Restaurați paleta originală ; Sfârșitul programului ; procedura read palette ; Plasează paleta VGA pe șir la ES:DI read palette mov mov afară rgos pag dx, C h al,O dx,al ; Port C h - index ; Începe de la zero DAC/mod culoare citind mov dl,OC h ; Port O C h - date DAC mov cx, * ; Citiți x octeți rep insb ; la șirul de la ES:DI ret citeste palette endp ; procedura write palette ; Încarcă o paletă în DAC VGA din șirul de la DS:SI write palate proc aproape mov dx,O C h ; Port O C h - Index/Mod DAC mov al, ; Începeți cu culoarea zero afară dx al mov dl, C h ; Port OZS IT - date DAC mov cx, * ; Scrieți x octeți rep outsb ; de la un șir la DS:SI ret write palette endp ; procedura dec palette ; Descrește valoarea fiecărui octet cu cu saturație (adică după ; octetul devine zero, nu mai este decrementat) de la șir la DS:SI ; și scrie rezultatul într-un șir în DS:SI dec palette proc aproape mov cx, * ; Lungimea șirului este de x octeți dec loop: lodsb i ; Citiți octetul test al,al ; Dacă este zero, jz deja zero ; sări peste următoarea comandă dec ax ; Decrementați octetul cu deja zero: stosb ; Scrie-o înapoi ; loop dec loop ; Repetați de x ori ret dec palette endp înregistrări Cum ; procedura walt retrace ; Așteptând să înceapă următoarea retracere a razei Programare la nivel de port wait retrace proc aproape push dix mov dx, DAh VRTL : inal,dx ; 'Port DAh - înregistrare ISR test al, jnz VRTL ; Așteptați sfârșitul retracerii curente VRTL : în al dx test al, ore jz VRTL ; Acum începeți următorul pop dx ret wait retrace endp palete: ; La sfârșitul programului păstrăm două exemplare ; palete - doar , Kb sfârşitul începutului Temporizator Până acum, tot ce știm despre temporizatorul de sistem este că declanșează o întrerupere IRQ de aproximativ , ori pe secundă De fapt, un temporizator de interval programabil este un sistem foarte complex format din trei părți - trei canale de cronometru, fiecare dintre acestea putând fi programat să funcționeze într-unul din șase moduri În plus, pe multe plăci de bază moderne există două astfel de temporizatoare, prin urmare, numărul de canale se dovedește a fi șase Pentru nevoile lor, programele pot folosi canalul (dacă nu au nevoie de difuzor) și canalul (dacă este prezent un al doilea cronometru) De asemenea, puteți reprograma canalul dacă este necesar, dar apoi va trebui să-l resetați, astfel încât BIOS-ul și DOS să poată continua să funcționeze În spațiul porturilor I/O, o zonă de la h la Fh este alocată cronometrului: □ portul h - canalul (generează IRQ ); □ portul h - canal (suporta upgrade de memorie); □ portul h - canalul (controlează difuzorul); □ portul h - registrul de control al primului cronometru; □ porturi h - h - al doilea timer al calculatoarelor cu magistrala Microchannel; □ porturi h - Bh - al doilea timer al calculatoarelor cu magistrala EISA Întregul control al temporizatorului se realizează prin ieșirea unui octet la de ore (pentru primul temporizator) Luați în considerare alocarea de biți în acest octet biții - : dacă nu este numărul canalului care trebuie programat , , - canalul , , bit - : - înghețați valoarea curentă a contorului pentru citire (în acest caz, biții - nu sunt utilizați) - citire/scriere doar octet mic - doar pentru citire/scriere octet mare - citire/scriere mai întâi octet mic, apoi octet mare Trucuri complexe de programare biții - : modul de funcționare canal OOO - întrerupeți IRQ când ajungeți la zero - multivibrator de așteptare - generator de impulsuri - generator de unde pătrate (mod principal) - un singur vibrator lansat de software - un singur vibrator declanșat hardware bit : format contor: - număr binar pe biți ( - OFFFFh) - număr BCD ( - ) Dacă biții - sunt AND, octetul trimis la h este considerat a fi o comandă de citire a contoarelor, al cărei format este diferit de comanda de programare a canalului: biții - : (citește codul de comandă a contoarelor) biții - : ce să citești: - mai întâi starea canalului, apoi valoarea contorului - valoarea contorului - starea canalului biții - : comanda se aplică canalelor - Dacă starea canalelor este solicitată cu această comandă, comenzile noi vor fi ignorate până când starea este citită de pe toate canalele care au fost ordonate de biții - Starea și valoarea contorului unui canal dat se obține prin citirea din portul corespunzător canalului dorit Formatul octetului de stare este următorul: bit : starea intrării OUTx în momentul în care comanda citire contoare a fost executată Deoarece contorul este decrementat cu la fiecare ciclu în modul , starea acestui bit, înghețată de comanda de înghețare a contorului de curent, va indica în ce jumătate de ciclu a fost cronometrul bit : / : starea contorului neîncărcat/încărcat (utilizat în modurile și și după o comandă de blocare a curentului) biții - : la fel ca biții - ai ultimei comenzi trimise la h Pentru a programa cronometrul în modul , care este ■ canalele și în mod implicit și cel mai frecvent utilizat în; programe, trebuie să faceți următoarele: Ieșiți comanda (pentru canalul ) b pentru a înregistra h, adică setați modul pentru canalul , iar la citire/scriere, va fi trimis mai întâi cuvântul inferior și apoi cel mai vechi Trimite octetul mic al valorii inițiale a contorului către portul corespunzător canalului selectat ( h pentru canalul ) Trimite octetul mare al valorii inițiale a contorului către același port După aceea, cronometrul va începe imediat să scadă numărul introdus de la valoarea inițială la zero cu o rată de de ori pe secundă (un sfert din viteza procesor ) De fiecare dată când acest număr ajunge la zero, va reveni la valoarea inițială din nou În plus, când contorul ajunge la zero, temporizatorul îndeplinește funcția corespunzătoare - canalul provoacă o întrerupere IRQ , iar canalul , dacă difuzorul este pornit, îi trimite începutul următoarei undă pătrată, forțându-l să funcționeze la set frecvență Valoarea inițială a contorului pentru canalul este implicită la OFFFFh ( ), care este maximul posibil Prin urmare, frecvența exactă de apelare IRQ este de / *= , ori pe secundă Pentru a citi valoarea curentă a contorului: Trimiteți către portul h o comandă pentru a îngheța valoarea contorului pentru canalul selectat (biții - sunt b) Trimiteți o comandă pentru a reprograma canalul fără a schimba modul de operare la portul h, dacă trebuie să utilizați o altă metodă de citire/scriere (de obicei nu este necesară) Citiți din portul corespunzător canalului selectat octetul scăzut al valorii contorului fix Citiți octeți mari de pe același port Există multe utilizări pentru cronometru, singura limitare aici este că cronometrul este o resursă globală și poate fi reprogramat doar pe sisteme multitasking cu cunoștințele sistemului de operare, dacă o permite deloc Ca exemplu, luați în considerare cum să utilizați un cronometru pentru a măsura cât de mult timp trece între o întrerupere hardware reală și momentul în care gestionarea acestei întreruperi primește controlul (pentru de ce este important acest lucru, consultați exemplele de programe de ieșire a sunetului din secțiunile și ) Deoarece IRQ apare atunci când valoarea contorului este zero, trebuie doar să îi citim valoarea în timpul pornirii handler-ului și să-i schimbăm semnul (deoarece „contorul cronometrului scade constant) ; latenţă asm ; Măsoară timpul mediu scurs între întreruperea hardware și pornire; manipulatorul corespunzător Imprimă timpul mediu în microsecunde după ; apăsând orice tastă (de fapt în / ) ; Programul folosește un adunator de biți pentru simplitate, astfel încât să poată da ; rezultate incorecte dacă așteptați mai mult de câteva minute / model minuscul cod org b; Pentru comanda shld ; Programul COM start: mov ax, b ; AH = h, AL = numărul de întreruperi int h; Aflați adresa managerului mov word ptr old int h, bx ; și scrieți-l în old int h mov word ptr old int h+ , es * mov aXi h ; AH' = h, AL = numărul de întreruperi Trucuri complexe de programare dx,offset int h handler ; DS:DX - adresa handlerului h mov int ; De acum înainte în variabila de latență mov int ah, h ; Setați handlerul, suma este acumulată ; Întrerupeți până când este apăsată orice tastă ax,word ptr latency word ptr counter, mov cmp ; Suma in AX ; Dacă tasta este apăsată imediat, jz dont divide ; evitați împărțirea la zero xor dx,dx ; Dx = div word ptr counter ; Împărțiți suma la numărul de economii dont divide: caii print ax ; și afișare pe ecran movax, h ; AH = h, AL = numărul de întreruperi Id-uri dx,dword ptr old int h ; DS:DX = adresa handlerului int h; Restaurați vechiul handler ret; Sfârșitul programului latență dw ; Cantitatea de întârzieri contor dw ; Numărul de apeluri întrerupte ; Gestionarea întreruperii h (IRQO) ; Specifică timpul scurs de când IRQO a fost declanșat int h handle r proc departe puști ax; Salvați carcasa folosită mov al, ; Fixarea valorii contorului în canalul afară h,al ; Port h: Registrul de control al temporizatorului ; Deoarece acest canal este inițializat de BIOS pentru citire/scriere pe biți, altele ; comenzile nu sunt necesare un al, h; Octet scăzut al contorului - mov ah, al ; în AN în al, h; Octetul mare al contorului din AL xchg ah,al ; Schimbă-le neg ax ; Schimbați-i semnul pe măsură ce contorul scade adăugați cuvântul ptr cs:latency,ax ; Adaugă la sumă inc word ptr cs:counter ; Măriți contorul de acumulare pop topor db OEAh ; comanda jmp far old int h dd ; Adresa vechiului handler int h handler endp ; procedura print ax ; Imprimă AX pe ecran în format hexazecimal print ax proc pagina xchg dx,ax ; DX = AX mov cx, ; Numărul de cifre de scos shift ax: - shld ax, dx, ; Obțineți următoarea cifră în AL rola dx, ; Ștergeți-l din OH și alții OFh; Lăsați doar acest număr în AL Programare la nivel de port LLL ± ^ cmp sbb das int -loop ret print ax endp Sfârşit al, oah' al, h h shift ax start ; Trei echipe de traduceri ; cifra hexadecimală în AL ; la codul ASCII corespunzător ; Ieșire pe afișaj ; Repetați pentru toate numerele Cronometrul poate fi folosit pentru a controla un difuzor, pentru a măsura cu precizie sincronizarea, pentru a crea întârzieri, pentru a controla comutarea procesului și chiar pentru a selecta un număr aleatoriu pentru a rula generatorul de numere aleatorii - valoarea curentă a contorului de canal de este o sămânță ideală pentru majoritatea aplicațiilor difuzor După cum este menționat în secțiunea , canalul al temporizatorului sistemului controlează difuzorul computerului - generează o undă pătrată cu o frecvență egală cu /counter start La programarea difuzorului, valoarea inițială a contorului temporizatorului se numește divizor de frecvență: se presupune că difuzorul funcționează la o frecvență de /divizor de hertzi După programarea canalului al temporizatorului, trebuie să porniți și difuzorul în sine Acest lucru se face prin setarea biților și ai portului h la Bitul activează de fapt acest canal de temporizator, iar bitul pornește difuzorul ;■Procedura de bip t ; Produce un sunet cu o frecvență de Hz (nota „mi” de octava mijlocie) ; Durată de / secundă pe difuzor Da Da proc aproape mov al, b; Canalul , modul afară h, al' mov al,ODh ; Octet scăzut al divizorului de frecvență D h afară h,al mov al, h; Octetul înalt al divizorului de frecvență afară h,al în al, h; Starea actuală a portului h în AL sau al, b; Setați biții și la afară h,al ; Acum difuzorul este pornit mov cx, h ; Cuvântul mare al numărului de microsecunde de pauză mov dx,' A h ; Cuvânt mic de pauză microsecunde mov ah, h; Funcția h: int' h; pauză în al, h şi al, b; Setați doi biți la zero out, h, al ; Difuzorul este acum oprit ret endp Trucuri complexe de programare Datorită omniprezenței plăcilor de sunet, difuzorul obișnuit pentru PC nu este acum practic folosit de nimeni sau este folosit pentru a emite mesaje de eroare Să revenim puțin mai târziu la sunet, dar deocamdată, amintiți-vă că în Secțiunea a fost luat în considerare un alt dispozitiv pentru determinarea orei și datei curente - un ceas în timp real Ceas în timp real și memorie CMOS Fiecare computer are un cip responsabil pentru menținerea datei și orei curente Pentru a se asigura că valorile nu sunt resetate de fiecare dată când alimentarea este oprită, microcircuitul are o zonă de memorie mică (de la la de octeți) realizată folosind tehnologia CMOS, ceea ce face posibilă reducerea la minimum a consumului de energie (de fapt, energia din astfel de circuite este cheltuită doar pentru încărcarea capacităților parazite atunci când starea se schimbă) celule de memorie) Cipul este alimentat de o baterie situată pe placa de bază și nu se oprește atunci când computerul este oprit Doar octeți de astfel de memorie nevolatilă sunt suficienți pentru a stoca ora reală, iar restul este folosit de BIOS pentru a stoca diverse informații necesare pornirii corecte a computerului Pentru comunicarea cu registrele CMOS și RTC, sunt alocate porturi I/O de la h la Fh, dar numai alocarea porturilor h și h este aceeași pentru toate plăcile de bază Port Și pentru scriere: index pentru selectarea registrului CMOS: bit : întrerupere NMI dezactivată pentru timpul de citire/scriere bit : index real Port h pentru citire și scriere: date CMOS După ce ați scris în portul h, trebuie să scrieți sau să citiți din portul h, altfel RTC va fi într-o stare nedeterminată Conținutul registrelor CMOS variază între BIOS-uri, dar primele de ore fac de obicei următoarele: OOh: RTC: secundă curentă ( - h sau - Bh) - format selectat de registrul OBh, BCD implicit Olh: RTC: secunde de alarmă ( - h sau - Bh sau OFFh (orice secundă)) h: RTC: minute curente ( - h sau - Bh) h: RTC: minute de alarmă ( - h sau - Bh sau FFh) h: RTC: ora curentă: - h/ - h (mod de ore) - h/ -ICh ( ore AM) lh- h/ - Ch (mod după-amiază de ore) h: RTC: ceas cu alarmă (la fel sau FFh dacă există vreo oră) h: RTC: ziua curentă a săptămânii ( - , este duminică) h: RTC : ziua curentă a lunii ( - h/ h-IFh) h: RTC: luna curentă ( - h/ -OCh) h: RTC: anul curent ( - h/ - h) Programare la nivel de port OAh: RTC: Registrul de stare A bit : - ceasul ocupat (actualizare în curs) biți - : divizor de fază ( - kHz - implicit) biții - : selectarea frecvenței de întrerupere periodică: - dezactivat UN - µs (minimum) - ms IT- , µs ( Hz) OBh: RTC: registru de stare B bit : actualizarea ceasului este dezactivată (setat înainte ca noile valori să fie scrise în registrele de dată și ceas) bit : întrerupere periodică a apelului (IRQ ) bit : întreruperea apelului de alarmă bit : întrerupeți apelul după finalizarea actualizării timpului bit : generarea undelor pătrate este activată bit : / : format de dată și oră binar/BCD bit : / : modul de ore/ ore bit O: Ora de vară automată în aprilie și octombrie Numai citire AXIS: RTC: registru de stare C bit : a apărut o întrerupere bit : întrerupere periodică activată bit : întrerupere de alarmă activată bit : întrerupere activată după finalizarea actualizării ceasului ODh numai citire: registru de stare D bit : pornire RTC/CMOS OEh: rezultatul operațiunii POST la ultima pornire a computerului: bit : RTC resetat din cauza lipsei de putere CMOS bit : suma de verificare a configurației CMOS invalidă bit : configurație nevalidă bit : dimensiunea memoriei nu se potrivește cu cea scrisă în configurație bit : eroare de inițializare a primului hard disk bit : ora RTC setată incorect (de exemplu, februarie) OFh: starea în care se afla computerul înainte de ultima repornire OOh - Ctr-Alt-Del, h - INT h, OAh, OBh, OCh - jmp, iret, retf la adresa stocată la h: h Alte valori indică faptul că reîncărcarea a avut loc în timpul POST sau în alte condiții neobișnuite h: tip de unități (biții - și - - tipurile primei și celei de-a doua unități) b - niciunul b - , Mb b- Kb OYOOL - , Mb b - , MB [ LJJ I Trucuri complexe de programare h: tip de hard disk (biții - și - sunt tipurile primului și celui de-al doilea hard disk, b dacă numărul tipului este mai mare de ) h: octet de stare hardware biții - : numărul de hard disk-uri instalate minus unu biți - : tip monitor ( , , , - EGA/VGA, x GGA, x CGA, MDA) bit : monitor prezent bit : tastatură prezentă bit : FPU prezent bit : unitate de disc prezentă h: octet mic din dimensiunea memoriei de bază în kiloocteți ( h) h: octet mare din dimensiunea memoriei de bază în kiloocteți ( h) h: octet mic de dimensiunea memoriei suplimentare (mai mare de MB) în kiloocteți h: octet mare de dimensiune de memorie suplimentară (mai mare de MB) în kiloocteți h: tipul primului hard disk dacă este mai mare de IAh: tipul celui de-al doilea hard disk dacă este mai mare de Eh: înregistrează suma de control octetul înalt h - Dh Fh: înregistrează suma de control octetul scăzut h - Dh h: octet mic de memorie suplimentară găsit la POST în kiloocteți h: octet mare de memorie suplimentară găsit la POST în kiloocteți h: primele două cifre ale anului în format BCD Datele de configurare stocate într-o zonă protejată de sumă de control sunt rareori necesare, iar pentru operațiuni simple de ceas și alarmă în timp real, este convenabil să utilizați întreruperea BIOS IAh Cu toate acestea, prin programarea RTC la nivel de port, puteți activa întreruperea periodică, un mod în care RTC declanșează IRQ la o rată stabilită, permițându-vă să lăsați IRQ pentru funcționarea sistemului dacă sunteți mulțumit de alegerea limitată a ratelor de întrerupere periodice De exemplu, să vedem cum se realizează citirea și scrierea în memoria CMOS, rtcti me asm Afișarea datei și orei curente din RTC modelul minuscul cod ; Pentru shr al, org h ; Programul COM start: mov al OBh ; CMOS OBh - registru de control B out h,al ; Port h - indice CMOS ■'Xi în al, h; Port h - date CMOS şi al, b; Setați bitul la zero (forma număr - BCD) afară h,al ; și scrie înapoi mov al, h; CMOS h - cele mai semnificative două cifre ale gfd cai print cmos ; Ieșire pe afișaj mov al, ; CMOS h - două cifre joase ale gyod'a Programare la nivel de port cai print cmos mov al,'-' ; Minus int h; Ieșire pe afișaj mov al, ; CMOS h - luna curentă cai print cmos mov al,'-~ ; Un alt minus int h mov al, ; CMOS h - zi cai print cmos mov al,' ' ; Spaţiu int h mov al, ; CMOS h - oră cai print cmos mov al 'ti' ; Litera „h” int h mov al,' ' ; Spaţiu int h mov al, ; CMOS h - minut cai print cmos mov al,':' ; Colon int> h mov al, Oh; CMOS OOh - secundă cai print cmos ret ; procedura print cmos ; Afișează conținutul celulei CMOS numerotate în AL ; Ia în considerare numărul citit din print cmos proc aproape afară h,al în al, h împinge toporul shr al, adăugați al 'O' int h pop topor şi al OFh adăugați al, h int h ret print cmos endp sfârşitul începutului CMOS este în format BCD ; Trimiteți AL către portul index CMOS ; Citiți datele ; Selectați cei patru biți înalți ; Adăugați codul ASCII pentru cifra ; Afișare pe ecran ; Extrageți cei patru biți inferiori ; Adăugați codul ASCII pentru cifra ; Afișare pe ecran Placi de sunet placi de sunet; compatibil cu anumite modele Sound Blaster arată ca patru dispozitive independente: □ DSP (Digital Signal Processor) - un dispozitiv care vă permite să scoateți și să citiți sunet digitizat; □ mixer (Mixer) - un sistem de control al volumului pentru toate canalele plăcii; oȚ G j GL! Trucuri complexe de programare □ FM (Frequency Modulation) sau AdLib (după numele primei plăci de sunet) - un dispozitiv care vă permite să sintetizați sunetul din unde sinusoidale și triunghiulare Cuvinte precum OPL sau OPL din descrierea plăcii sunt numerele de versiune ale sintetizatorului FM folosit; □ MIDI (Music Instrumental Digital Interface) este o interfață standard de transfer de date în echipamente muzicale Dar, în cazul nostru, este considerat GMIDI (MIDI generalizat) - un sistem de generare de muzică mai bun, care utilizează nu semnale sinusoidale artificiale, ci mostre de sunet ale diferitelor instrumente Din păcate, calitatea acestor mostre în majoritatea plăcilor ieftine lasă de dorit Numerele portului I/O care oferă acces la toate aceste dispozitive se bazează pe portul de bază, care este de obicei b, dar sunt permise și configurații cu h, h h, h, h și h În plus, interfața GMIDI folosește o serie diferită de porturi care pot începe fie cu h, fie cu h În descrierile porturilor, vom presupune că cele de bază sunt h și h Zona de porturi de interfață AdLib începe la h Există un număr mare de modificări ale plăcilor Sound Blaster, care diferă, printre altele, în setul de comenzi acceptate și porturi I/O După numele fiecărei comenzi sau port, vom indica numele abreviat al plăcii de la care este suportată această comandă sau port: □ SB - Sound Blaster ; □ SB - Sound Blaster ; □ SBPro - Sound Blaster Pro; □ SBPro - Sound Blaster Pro ; / □ SB - Sound Blaster ; □ ASP - Sound Blaster ASP; □ AWE - Sound Blaster AWE Programare DSP Procesorul digital este cea mai importantă parte a plăcii de sunet Cu ajutorul acestuia se realizează ieșirea sunetului obișnuit digitizat, precum și înregistrarea sunetului dintr-o sursă externă într-un fișier Pentru activitatea sa, pe lângă porturile descrise în această secțiune, DSP utilizează întreruperi și un controler DMA Programarea DMA cu un exemplu de program care îl folosește pentru a reda sunet va fi tratată în secțiunea DSP este deservit prin următoarele porturi: h pentru scriere: resetare DSP (SB)' Scrierea pe acest port realizează o reinițializare completă a DSP-ului, întrerupând toate procesele în curs Operația de resetare a DSP trebuie efectuată cel puțin o dată după o repornire a sistemului pentru ca acesta să poată fi utilizat Procedura de resetare este următoarea: Numărul este scris pe portul h (începutul inițializărilor) Se menține o pauză de cel puțin , μs Programare la nivel de port Numărul (sfârșitul inițializării) este scris pe portul h O pauză este menținută pentru maximum µs În timpul pauzei, puteți citi portul Eh Când bitul este setat în numărul citit (datele sunt gata), puteți trece imediat la pasul În caz contrar, este logic să repetați procedura folosind un alt port de bază Citirea din portul Ah Dacă numărul citit este OAAh, DSP-ul a fost inițializat cu succes În caz contrar, puteți „reveni la pasul , dar după µs după scrierea la h, va fi sigur să spuneți că DSP-ul cu adresa de bază h nu există sau nu funcționează Ah pentru citire: citirea datelor de la DSP (SB) Citirea de pe acest port este folosită pentru a transfera toate datele posibile de la DSP către programe Procedura de citire constă în două etape: Efectuați un ciclu de citire al portului Eh până când bitul al octetului de citire este egal cu unu Citiți din portul Ah Ch pentru a scrie: scrieți date și comenzi DSP (începând de la SB) Acest port unic este folosit pentru a transmite întregul set de comenzi DSP și pentru a trimite date (argumente de comandă) către acesta Procedura, inregistrari: Efectuați un ciclu de citire portului Ch până când bitul al octetului de citire este zero Scrieți pe portul Ch Ch pentru a citi: DSP pregătit pentru a primi comandă (SB) Dacă, la citirea de pe acest port, bitul este resetat la zero, DSP-ul este pregătit să primească următorul octet către portul Ch pentru scriere Semnificația biților rămași este nedefinit Eh de citit: DSP gata să trimită date (începând de la SB) Dacă, la citirea de pe acest port, octetul este setat la unu -■ DSP este gata să transfere următorul octet prin portul Ah Eh pentru citire (același port): confirmarea procesării întreruperii pe biți (SB) Un handler de întrerupere generat de placa de sunet la sfârșitul unei operațiuni pe biți trebuie să efectueze neapărat o citire de pe acest port înainte de finalizare (în plus față de procedura obișnuită pentru trimiterea unui semnal EOI către controlerul de întrerupere corespunzător) Fh de citit: confirmare procesare întrerupere pe biți (SB ) Un handler de întrerupere generat de placa de sunet la sfârșitul unei operațiuni pe biți trebuie să efectueze o citire de pe acest port înainte de finalizare (în plus față de procedura normală de trimitere a unui semnal EOI către controlerul de întrerupere corespunzător) Acum să ne uităm la comenzile DSP Toate sunt trimise la placa de sunet prin portul Ch, așa cum este descris mai sus Comanda poate fi urmată de argumente, Trucuri complexe de programare care sunt transmise în același mod (inclusiv în așteptarea ca o comandă să fie gata) L h: stare DSP (învechit) (SB - SBPro ) Returnează informații despre funcționarea curentă a DSP: bit : difuzorul activat bit : ADC stereo activat, ■ bit : întotdeauna bit : redare directă PCM pe biți bit : redare ADPCM pe biți peste DMA bit : redare ADPCM pe , biți peste DMA bit : redare ADPCM pe biți peste DMA bit : redare în curs -bit PCM prin DMA h, NN: Redare directă a sunetului pe biți (SB) Emite următorul octet (NN) din audio digitizat pe biți necomprimat pentru redare Când utilizați această metodă de redare, programul în sine trebuie să se asigure că datele noi sunt întotdeauna gata (adică nu sunt citite de pe disc în timpul funcționării) și că octeții sunt trimiși către DSP la frecvența necesară (Frecvențele de până la kHz sunt acceptate în acest mod ) Procedura de ieșire este simplă: Ieșiți comanda h și următorul octet de la digitizare la DSP Așteptați timpul necesar și reveniți la punctul Pentru a transfera octeți la o anumită frecvență, este obișnuit să reprogramați cronometrul sistemului, așa cum va fi arătat la sfârșitul acestei secțiuni Dar din cauza limitărilor de calitate a sunetului și a consumului mare de resurse, această metodă de redare practic nu este utilizată h, LO, HI: Redare directă PGM pe biți prin DMA (SB) Pornește procesul de redare a datelor la care este setat canalul DM corespunzător A (vezi secțiunea ): Instalați un handler de întreruperi de pe placa de sunet (și activați-l în controlerul de întreruperi) Executați comanda h sau setați în alt mod rata de eșantionare Rulați comanda ODlh (activați dinamica Configurați DMA (mod h + numărul canalului) Rulați comanda h Argumentele IO și HI sunt octeții mici și mari ai lungimii secțiunii redate, minus unu De la manipulatorul de întreruperi, confirmați-l citind portul ёb și trimițând octetul h la controlerul de întrerupere corespunzător Rulați comanda D h (opriți difuzorul) La plăcile care încep de la SB , se recomandă utilizarea comenzilor C?h pentru acest mod Programare la nivel de port ! h, LO, HI: Redare directă ADPCM pe biți prin DMA (SB) Pornește procesul de redare a datelor similar cu comanda h, dar trebuie să fie stocată în format comprimat Creative ADPCM pe biți Lungimea specificată ca argumente pentru această comandă este (număr octeți + )/ Valoarea utilizată de ultima comandă de ore este folosită ca octet nul în procedura de despachetare ADPCM În caz contrar, procedura de redare este aceeași cu comanda h h, LO, W: redare directă a ADPCM pe biți prin DMA cu noul octet zero (SB) La fel ca h, dar primul octet al datelor va fi tratat ca un octet nul pentru procedura de despachetare ADPCM ICh: Redare PCM pe biți prin DMA cu inițializare automată (SB ) Începe redarea cu inițializare automată - cel mai bun mod oferit de plăcile de sunet În acest mod, DSP-ul trece în buclă conținutul zonei de memorie specificate, revenind instantaneu la început până când este oprit de o comandă ODAh sau de o nouă comandă de redare DMA Întregul secret constă în faptul că placa generează o întrerupere nu doar când ajunge la capătul blocadei, ci și când ajunge la mijloc Astfel, în timp ce DSP-ul redă a doua jumătate a bufferului, putem citi următorii câțiva kiloocteți din prima jumătate fără a opri redarea pentru un moment f Instalați dispozitivul de gestionare a întreruperii plăcii de sunet și activați-l în controlerul de întrerupere Executați comanda h sau setați în alt mod rata de eșantionare Rulați comanda h (setați dimensiunea bufferului DMA - (număr octeți + )/ - ) Executați comanda ODlh (porniți difuzorul) Setați DMA (mod h + numărul canalului) Executați comanda ICh În manipulatorul de întreruperi: completați următoarea jumătate a bufferului În manipulatorul de întreruperi: confirmați întrerupere citind de la Eh și scriind h la controlerul de întrerupere Așteptați până când toate datele au fost redate Rulați comanda D h (opriți difuzorul) I Executați comanda ODOh (stop -bit DMA transfer) Executați comanda ODAh (încheierea modului de auto-inițializare) Executați comanda ODOh (opriți transferul DMA pe biți) La plăcile care încep cu SB , se recomandă utilizarea comenzilor C?h pentru acest mod , IFh: redare ADPCM pe biți prin DMA cu inițializare automată (SB ) I Trucuri complexe de programare Similar cu comanda ICh, dar datele sunt stocate în format ADPCM pe biți cu un octet nul Lungimea blocului se calculează după cum urmează: lungime = (număr octeți + )/ + , lungime bloc = (lungime + )/ - h: citire directă a datelor pe biți de la ADC (SB) Comanda este destinată citirii sunetului digitizat dintr-o sursă externă Se utilizează următoarea procedură: Rulați comanda b Citiți octetul următor Așteptați timpul necesar și reveniți la punctul Problemele cu această comandă sunt exact aceleași ca și cu IOB h, LO, HI: Citiți PCM pe biți prin DMA (SB) Un analog al comenzii h, dar nu redă, ci înregistrează sunet Secvența operațiilor este identică cu cazul b, dar modul DMA folosit este - b + numărul canalului Ch: Scrieți PCM pe biți prin DMA cu inițializare automată (SB ) Un analog al comenzii СЬ, dar nu redă, ci înregistrează sunet Secvența de acțiuni este identică cu cazul ІСЬ, dar modul DMA utilizat este - b + numărul canalului h: Citire MIDI directă (SB) Citește următorul eveniment MIDI: Rulați comanda h Citiți un eveniment MIDI (până la de octeți) b: Citire MIDI cu întrerupere (SB) Permite generarea unei întreruperi de pe placa de sunet atunci când sosește un nou eveniment MIDI Pentru asta ai nevoie de: Instalați un handler de întrerupere Executați comanda b În manipulatorul de întreruperi: citiți evenimentul MIDI În manipulatorul de întreruperi: confirmați întreruperea citind de la EB și scriind b la controlerul de întrerupere Executați din nou comanda b pentru a anula generarea întreruperilor b: eveniment MIDI citit direct cu delta time (SB) Citește următorul eveniment MIDI și timpul delta pe de biți, adică timpul în microsecunde de la coexistența anterioară MIDI (Datele sunt citite în următoarea ordine: octet scăzut de timp, octet mediu de timp, octet mare de timp, comandă MIDI ) Este sub forma unei secvențe de evenimente MIDI, fiecare dintre acestea fiind precedat de un timp delta , acea muzică este înregistrată în fișiere MIDI Programare la nivel de port h: Read Delta Time MIDI Event with Interrupt (SB) Activează/dezactivează generarea de întreruperi de pe placa de sunet, similar comenzii h, dar când sunt citite, evenimentele MIDI sunt transmise împreună cu timpii delta, ca în comanda h de ore: modul de acces direct UART (SB ) Dezactivează DSP, după care toate comenzile de scriere/citire pe porturile sale (folosind același mecanism de pregătire) sunt tratate ca evenimente MIDI Singura modalitate de a scoate DSP-ul din acest mod este printr-o reinițializare completă h: mod de acces direct UART cu întrerupere (SB ) Comută porturile DSP la UART similar cu comanda h, dar de fiecare dată când un nou eveniment MIDI este gata de citit, este apelată o întrerupere a plăcii de sunet h, MIDI: înregistrare directă MIDI (SB) Trimite un eveniment MIDI h, TC: setați constanta de timp (SB) Setează rata de eșantionare folosind o constantă de un octet calculată după cum urmează: TC - - ( / (număr de canale x frecvență)), unde numărul de canale este pentru mono și pentru stereo h, Yu, HI: setați rata de eșantionare (SB ) Similar cu h, dar este indicată valoarea adevărată a frecvenței (mai întâi octeții mici, apoi cei înalți) Numărul de canale este determinat automat Frecvența reală este totuși rotunjită la cea mai apropiată valoare TC posibilă h: Reluare redare DMA oprită pe biți (SB ) Reia redarea DMA pe biți oprită de ODAh cu inițializare automată h: Reluare redare oprită pe biți prin DMA (SB ) Reia redarea audio pe biți prin DMA oprită prin comanda D h cu inițializare automată h, LO, HI setați dimensiunea tamponului DM A(SB ) Setează numărul de octeți minus unu pentru următoarea comandă de transfer DMA (octetul mic mai întâi, apoi octetul mare) h, IX), HI: Redare directă ADPCM pe biți prin DMA (SB) Similar cu h, dar utilizează versiunea pe biți a formatului Creative ADPCM h, IX), HI: redare directă a ADPCM pe biți prin DMA cu un nou octet nul (SB) Similar cu h, dar folosește versiunea pe biți a formatului Creative ADPCM j lZJJ Trucuri complexe de programare h, LO, HI; redare directă a ADPCM pe , biți prin DMA (SB) Similar cu h, dar utilizează varianta de , biți a formatului Creative ADPCM h, LO, HI; redare directă a ADPCM pe , biți prin DMA cu noul octet zero (SB) Similar cu h, dar folosește versiunea pe , biți a formatului Creative ADPCM Dh: Redare ADPCM pe biți prin DMA cu inițializare automată (SB ) Similar cu IFh, dar utilizează varianta pe biți a formatului Creative ADPCM Fh; redarea ADPCM pe , biți prin DMA cu inițializare automată (SB ) Similar cu IFh, dar utilizează varianta de , biți a formatului Creative ADPCM h, LO, HI: sunet DSP (SB) Ieșiți numărul specificat de octeți de tăcere la rata de eșantionare curentă h B?h/ C?h MOD, LO, HI: interfață generică la DSP (SB ) Comenzile B?h sunt folosite pentru operațiuni pe biți, comenzile C?h sunt pentru cele pe biți Cei patru biți inferiori determină modul: bit : întotdeauna bit : FIFO folosit bit : este utilizată auto-inițializarea DMA bit : direcția de transfer ( - redare, - digitizare) Argumentele acestei comenzi sunt mode, octet mic de lungime, octet mare de lungime (nu este nevoie să setați dimensiunea bufferului DMA înainte de comanda specificată) Doar doi biți sunt definiți în byte de mod (restul trebuie să fie zero): bit : datele sunt tratate ca numere semnate bit : modul stereo " Lungimea este în toate cazurile numărul de octeți minus unu pentru operațiunile pe biți și numărul de cuvinte minus unu pentru operațiunile pe biți ODOh: opriți operațiunea DMA pe biți (SB) Oprește o operațiune DMA simplă (fără auto-inițializare) pe biți ODlh: activați difuzorul (SB) Permite să funcționeze ieșirea difuzoarelor (difuzoare etc ) După o resetare DSP, acest canal este dezactivat D h: opriți difuzorul (SB) Dezactivează ieșirea difuzoarelor (difuzoare etc ) D h: continuă operațiunea DMA pe biți (SB) Continuă operațiunea DMA oprită de comanda ODOh D h: opriți operarea DMA pe biți (SB) Oprește o operațiune DMA simplă (fără auto-inițializare) pe biți D h: continuă operațiunea DMA pe biți (SB) Continuă operațiunea DMA oprită prin comanda D h Programare la nivel de port unsprezece D h: Determinați starea difuzorului (SB): Returnează h dacă difuzorul este oprit; OFFh dacă este activat D h: Încheierea operației DMA pe biți cu auto-inițializare (SB ) Această comandă încheie operațiunea numai după ce s-a terminat redarea blocului curent Pentru a opri imediat redarea, trebuie să executați comenzile D h, D h, D h și D h în secvență ODAh: operațiune DMA completă pe biți cu auto-inițializare (SB ) Similar cu D h, dar pentru operațiuni pe biți OEOh, BYTE: verificați DSP pe acest port (SB ) Orice octet trimis ca argument la această comandă este returnat atunci când este citit de la DSP ca complement pe biți (DSP-ul efectuează o operație NOT pe el) OElh: detectarea numărului versiunii DSP (SB) Returnează versiunile majore și minore ale DSP-ului în secvență: -SB -SB , g-SBPro ?-SBPro ? -SB - SB SCSI- - AWE E h: citire, Copyright DSP (SBPro ) Returnează un șir ASCIZ cu informațiile despre Copyright ale plăcii E h, BYTE: scrieți în registrul de testare (SB ) Plasează un octet într-un registru special neutilizat care este reținut chiar și după reinițializarea DSP-ului E h: citit din registrul de testare (SB ) Returnează octetul plasat anterior în registrul de testare prin instrucțiunea E h OFOh: generarea formei de undă sinusoidală (SB) Determină ca DSP să redă o undă sinusoidală la aproximativ kHz, care poate fi întreruptă doar prin resetarea DSP-ului F h: Cerere de întrerupere în modul pe biți (SB) Generează o întrerupere de pe placa de sunet Este de așteptat ca handlerul să citească din portul Eh ca confirmare F h: cerere de întrerupere în modul pe biți (SB) Generează o întrerupere de pe placa de sunet Este de așteptat ca handlerul să citească din portul Fh ca confirmare OFBh: statut DȘP (SB ) Returnează octetul de stare al operațiunii DSP curente: bit : redare pe biți prin DMA bit : citire pe biți prin DM A j ^T TTP Trucuri complexe de programare bit : redare pe biți prin DMA bit : citire pe biți prin DMA ' bit : difuzorul activat biții - : nu sunt definiți bit : TC modificat (poate fi zero dacă comanda anterioară de de ore a încercat să seteze o frecvență neacceptată) OFCh: informații suplimentare (SB ) Returnează un octet de stare suplimentar pentru operațiunea DMA curentă: bit : modul sincron (înregistrare și redare simultană) bit : mod pe biți cu auto-inițializare bit : mod pe biți cu auto-inițializare OFDh: ultima comandă executată (SB ) Returnează ultima comandă DSP reușită Programarea mixerului Acest dispozitiv este conceput pentru a controla volumul pe toate canalele utilizate de placa de sunet Mixerul este controlat de doar două porturi I/O h pentru înregistrare: selecția registrului mixerului (SBPro) Scrierea pe acest port selectează numărul de înregistrare care va fi accesat la apelurile ulterioare către portul h h de citire/scriere: registru mixer de citire/scriere (SBPro) Citirea și scrierea pe acest port are ca rezultat citirea și scrierea în registrul mixerului corespunzător Luați în considerare scopul lor Înregistrați OOh pentru scriere: resetare și inițializare (SBPro) Selectarea acestui registru la portul h începe inițializarea Ar trebui să așteptați cel puțin µs, apoi să scrieți numărul Olh pe portul h (comanda „inițializare completă”), Înregistrați-vă pentru a citi: starea mixerului (SBPro) Returnează ultimul număr de registru Registrul de citire/scriere h: nivel DAC (SBPro) biții- - : nivelul DAC drept biții - : nivelul DAC din stânga Registrul OAh pentru citire și scriere: nivel microfon (SBPro) biții - : nivelul microfonului Înregistrare ore pentru citire și scriere: nivel general (SBPro) biții - : nivel comun dreapta biții - : nivel comun stânga Programare la nivel de port eu Înregistrați de ore pentru citire și scriere: nivel FM (SBPro) biți - : nivel FM dreapta biți - : nivel FM stânga Registrul de citire/scriere h: nivel audio CD (SBPro) biți - : nivel dreapta audio CD biți - : nivel stânga audio CD Registrul de citire/scriere Eh: Nivel de intrare linie (SBPro) Biți - : Nivel de intrare linie din dreapta Biții - : Nivel de intrare linie din stânga Înregistrați de ore pentru citire și scriere: nivel comun stânga (SB ) biții - : nivel comun stânga f Registrul de citire/scriere h: Nivel comun dreapta (SB ) Biții - : Nivel comun dreapta Registrul de citire/scriere h: DAC Left Level (SB ) Biții - : DAC Left Level Înregistrați de ore pentru citire și scriere: nivel drept DAC (ȘB ) biții - : nivel dreapta DAC Registrul de citire/scriere h: nivel FM stânga (SB ) biți - : nivel FM stânga Registrul de citire/scriere h: nivel FM dreapta (SB ) biți - : nivel FM dreapta Registrul de citire/scriere b: nivel stânga audio CD (SB ) biți - : nivel stânga audio CD Registrul de citire/scriere h: nivelul dreptului audio CD (SB ) biții - : nivelul dreptului audio al CD-ului Registrul de citire/scriere h: Linie de intrare Left Level (SB ) Biții - : Line-In Left Level Registrul de citire/scriere h: Nivel de intrare linie dreapta (SB ) Biții - : Nivel de intrare de linie dreapta Registrul de citire/scriere: Mic Level (SB ) biți - : Mic Level Registrul de citire/scriere BR: Nivel difuzor PC (SB ) biți - : Nivel difuzor PC Registrul cu canale pentru citire și scriere: control ieșire (SB ) bit : microfon activat bit : canal audio CD activat bit : canal audio CD stânga activat J LLTT Trucuri complexe de programare bit : canalul drept de intrare de linie activat bit : canalul stânga de intrare de linie activat biții - : zerouri Registrul de citire/scriere Dh: Control canal de intrare din stânga (SB ) Bit : Microfon activat bit : canalul drept audio CD activat bit : canalul stânga audio CD activat bit : canalul drept de intrare de linie activat bit : canalul stânga de intrare de linie activat bit : canal FM dreapta activat bit : canalul FM a lăsat pornit bit : zero - Registrul de citire/scriere SHE: Control canal de intrare dreapta (SB ) Bit : Microfon activat bit : canalul drept audio CD activat bit : canalul stânga audio CD activat bit : canalul drept de intrare linie activat h bit : canalul stânga de intrare de linie activat bit : canal FM dreapta activat bit : canalul FM a lăsat pornit bit : zero Registrul de citire/scriere Hi: Left Input Gain Level (SB ) Biții - : Left Input Gain Registrul de citire/scriere h: Nivel de amplificare de intrare dreapta (SB ) Biții - : Câștig de intrare din dreapta Registrul de citire/scriere b: Left Output Gain Level (SB ) Biții - : Left Output Gain Registrul de citire/scriere h: Nivel de câștig de ieșire din dreapta (SB ) Biții - : Câștig de ieșire din dreapta Registrul de citire/scriere h^ Control automat câștig (SB ) Bit : Control automat câștig activat Registrul de citire/scriere h: Left Treble Boost Level (SB ) biți - : Left Treble Gain Registrul de citire/scriere h: Nivel de creștere a înaltelor drepte (SB ) biți - : Amplificare a înaltelor la dreapta - Registrul de citire/scriere h: nivelul de amplificare a basului la stânga (SB ) biți -^i: amplificare a basului la stânga Registrul de citire/scriere h: Nivel de amplificare a basului drept (SB ) Biții - : Amplificare a basului la dreapta Programare la nivel de port Registrul de citire/scriere h: selectare întrerupere (SB ) bit : IRQ bit : IRQ ht :IRQ ht :IRQ biții - : întotdeauna Această setare este păstrată atunci când mixerul este resetat sau chiar când computerul este pornit la cald Registrul de citire/scriere h: Selectare DMA (SB ) bit : DMA pe biți bit : DMA pe biți bit : bit : -bit DMĂ bit : bit : DMA pe biți bit : DMA pe biți „bit : DMA pe biți Această setare este păstrată atunci când mixerul este resetat sau chiar când computerul este pornit la cald Citiți Registrul h: Stare întrerupere (SB ) bit : întreruperea de biți este anulată bit : întrerupere gestionată de operare pe biți bit : procesarea întreruperii operațiunii MPU- în curs biți - : rezervat Sinteză de frecvență (programare AdLib) Sintetizatorul, situat pe placa de sunet și responsabil pentru muzica FM, este controlat de trei porturi - un port de stare, un port de selectare a registrului și un port de date Pentru compatibilitate cu diferite plăci, acestea sunt duplicate de mai multe ori h, h Citire: Port de stare FM (SB) Citire de h: Port de stare FM stânga (SBPro) h, Ah Citire: Port de stare FM dreapta (SBPro) Stabilește dacă cronometrele FM au expirat și care biții - : întotdeauna bit : temporizatorul (perioada - µs) a fost declanșat bit : temporizatorul (perioada - µs) a fost declanșat bit : unul dintre cronometre s-a oprit h, h pentru scriere: selectare registru FM (SB) h pentru înregistrare: selecția registrului canalului stâng FM (SBPro) h, Ah pentru înregistrare: selecția registrului canalului drept FM (SBPro) Scrierea unui număr pe acest port selectează ce registru al sintetizatorului va funcționa cu comenzile de scriere ulterioare în portul de date g şi FITlJ Trucuri complexe de programare Procedura de scriere a unui număr în registrul FM este următoarea: Scrieți în portul de selecție a registrului Așteptați cel puțin , µs Scrieți în portul de date Așteptați cel puțin µs înainte de orice altă operațiune pe placa de sunet Pe plăcile moderne, aceste pauze pot fi reduse Deci, dacă se folosește sintetizatorul OPL , valoarea de pauză necesară este , și , µs h, h pentru a scrie: scrie pentru a înregistra FM (SB) h pentru a scrie: scrieți în registrul canalului din stânga FM (SBPro) h, Bh pentru a scrie: scrieți în registrul canalului drept FM (SBPro) În total, de registre sunt disponibile în aceste sintetizatoare - de la Olh la F h Să considerăm cele mai utile Registrul Olh: registru de testare biții - : bit : Cipul FM controlează forma de undă biții - : Înregistrare h: contor pentru prima dată, Valoarea acestui contor este incrementată cu unul la fiecare µs Când registrul depășește, întreruperea temporizatorului IRQ este generată și biții și sunt setați în portul de stare Înregistrare h: al doilea cronometru Valoarea acestui contor este incrementată cu unul la fiecare µs Când registrul depășește, întreruperea temporizatorului IRQ este generată și biții și sunt setați în portul de stare Register h: Registrul de control al temporizatorului bit : setarea bitului pornește primul timer (cu valoarea inițială a contorului plasată în registrul h) bit : setarea bitului pornește al doilea cronometru (cu valoarea inițială a contorului plasată în registrul h) biții - : rezervați bit : al doilea timer mask - setarea acestui bit dezactivează al doilea timer când bitul este setat bit : first timer mask - setarea acestui bit dezactivează primul timer când bitul este setat bit : resetează steaguri pentru ambele cronometre Înregistrare h: Activați modurile CSM și Keysplit biții - : rezervat bit : activați modul de împărțire a tastelor bit : - modul CSM; - modul FM Programare la nivel de port Modul CSM (sine wave speech synthesis) a fost folosit pe plăcile AdLib fără DSP pentru a reproduce sunetul digitizat (cu o calitate foarte slabă) Următoarele registre sunt descrise în grupuri - câte unul pentru fiecare voce Înregistrează h - h: setări diferite pentru modul f biții - : care armonică va produce sunet (sau modulație) în raport cu frecvența setată cu precizie a vocii: - cu o octava mai jos - cu o frecvență de voce setată cu precizie - cu o octava mai sus - o octavă și o cincime mai sus - două octave mai sus - două octave și o treime majoră deasupra - două octave și o cincime mai mare - două octave și o șapte mică deasupra - trei octave mai sus - trei octave și o secundă mare deasupra: A, B - trei octave și o treime majoră deasupra ' C, D - trei octave și o cincime mai sus E, F - trei octave și o șapte majoră deasupra ritmului : modul de scalare a tastaturii - dacă este selectat, lungimea sunetului se scurtează și crește bit : prelungirea etapei de susținere a sunetului (adică nu se estompează imediat după ce crește) bit : pornește modul vibrato (adâncimea acestuia este controlată de un steag în registrul OBDh) bit : aplicați modulația de amplitudine (adâncimea sa este controlată de un steag în registrul OBDh) Registre h - h: controlează nivelurile de ieșire, biții - : nivelul de ieșire total al canalului ( - maxim, - minim) biții - : nivelul de ieșire în jos cu frecvența crescută: - nu retrogradați - , dB pe octava - dB pe octava - dB pe octava Înregistrează h - h: Rata de creștere/cădere Biții - : Rata de cădere ( - minim, F - maxim) Biții - : Rata de creștere ( - minim, F - maxim) Registre h - h: Nivel suport/rampa de eliberare biți - : rampă de eliberare ( - minim, F - maxim) biți - : nivel suport ( - minim, F - maxim) Legea J ii II LII Trucuri complexe de programare Înregistrează OAOh - Â h: notă (octet scăzut) biții - : biții - ai numărului notei Înregistrează OBOh - B h: octavă și notă (biți cei mai semnificativi), biții - : biții - ai numărului notei biții - : număr de octavă bit : dezactivați sunetul acestui canal Valoarea numărului notei de biți corespunde următoarelor note reale (pentru octava a patra): Bh - C# ( , Hz) h - D ( , Hz) h-D# ( , Hz) OlBOh - E ( , Hz) OlCAh - F ( , Hz) E h - F# ( , Hz) - G ( , Hz) h - G# ( , Hz) lh - A ( , Hz) h - A# ( , Hz) h - V ( , Hz) AEh - C ( , Hz) Registrele OCOh - C h: feedback/algoritm • • bit : - primul operator îl modulează pe al doilea, - ambii operatori produc direct sunet biții - : puterea feedback-ului Dacă acest câmp este diferit de zero, prima instrucțiune va trimite înapoi o fracțiune din semnalul de ieșire; în tine însuți Register OBDh: adâncimi de modulație/ritm bit : HiHat activat bit : Chimbal activat bit : Tam-tain activat beat : Snare tobe activată beat : Bass toba activată bit : - ritm activat ( voci per melodie), - ritm dezactivat ( voci per melodie) bataie : profunzime vibrato bit : adâncimea modulării în amplitudine ( - , dB; - dB) Înregistrează OEOh - F h: selecția formei de undă biții - : formă de undă utilizată când bitul al registrului Olh este setat la : sinusoid : undă sinusoidală fără jumătate negativă : valoarea absolută a undei sinusoidale (semiundele inferioare sunt reflectate în sus) I: pulsuri din dinți de ferăstrău Programare la nivel de porturi ІІМВШМШ Pentru a extrage un sunet simplu din FM, efectuați următoarea secvență de acțiuni: / * Scoateți la zero toate registrele (o modalitate brută de a inițializa AdLib) Canalul va fi folosit pentru voce: Scrieți numărul Olh în registrul h - multiplicitatea modulației este Scrieți numărul h în registrul h - nivelul de modulație este de dB Scrieți numărul OFOh în registrul h - creșterea este rapidă, căderea este lungă Să scriem numărul h în registrul h - sprijin și eliberează Media Scrieți numărul h în registrul OAOh - nota D# Canalul va fi utilizat pentru operatorul de transport: Scrieți în registrul h numărul Olh - multiplicitatea purtătorului Scrieți în registrul h numărul OOh - volumul maxim pentru purtător ( dB) Scrieți în registrul h numărul OFOh - creșterea este medie, declinul este lung Să scriem numărul h în registrul h - suport și lansare - medie I Scrieți numărul h în registrul OBOh - setați octava, biții înalți note și porniți vocea Din acest moment se aude sintetizatorul Scrieți numărul h (sau orice număr cu un bit iulie ) în registrul OBOh pentru a opri sunetul Exemplu de program Programarea plăcilor de sunet moderne este o sarcină foarte dificilă, așa că, ca exemplu, să luăm în considerare o operație frecvent utilizată - redarea sunetului digitizat În acest scop, va trebui programat doar DSP-ul Pentru a scoate sunetul printr-o placă de sunet, se poate folosi unul dintre cele trei moduri: ieșire directă (comandă h), când programul însuși trebuie să trimită octeți individuali de la sunetul digitizat la DSP la frecvența necesară; modul DMA simplu, când este scos un bloc de date, după care este apelată o întrerupere; și DMA cu auto-inițializare, când datele sunt scoase în mod continuu și este apelată o întrerupere după ieșirea fiecărui bloc În această ordine crește calitatea sunetului reprodus Deoarece încă nu știm cum să lucrăm cu DMA, vom lua în considerare prima metodă Pentru a scoate date digitizate la frecvența dorită către DSP, va trebui să reprogramați canalul al temporizatorului de sistem la frecvența dorită și să instalați propriul dvs handler de întrerupere h Acest lucru va perturba funcționarea ceasului sistemului, deși nu puteți dezactiva gestionarea complet veche, ci îi puteți transfera controlul de aproximativ , ori pe secundă, adică, în special, cu fiecare apel către handlerul nostru la o frecvență de Hz Vă vom arăta cum să faceți acest lucru cu un program simplu care va reda c:\windows\media\tada wav (sau c:\windows\tada wav dacă modificați directiva EQU corespunzătoare la începutul programului) exact in acest fel ; wavdir ăsfn ; Redă fișierul c:\windows\media\tada wav fără a utiliza DMA ; Funcționează în mod normal numai sub DOS în modul real Trucuri complexe programate! ; (adică nu într-o fereastră DOS (Windows) și nu sub EMM , QEMM sau alte ; programe similare) / FILESPEC equ "din: \windows\media\tada wav" ; Nume fișier tada wav c calea completă (înlocuiți cu c:\windows\tada wav ; pentru versiunile mai vechi de Windows) SBPORT equ h , ' ; Portul de bază al plăcii de sunet (înlocuiți, ; dacă al tău este diferit) modelul minuscul cod ; Pentru push/pop org lOOh start: ; Programul COM cai dsp reset ; Resetați și inițializați DSP-ul jc no Master mov bl, OD h ; Comanda DSP D h cai dsp write ; Activați sunetul caii open file ; Deschideți și citiți tada wav caii hook int - Conectați o întrerupere a temporizatorului mov bx, ; Timer divider pentru frecventa Hz ; (corespunde de fapt la Hz) caii reprogram pit ; Reprogramați cronometrul bucla principala: ; Ciclul principal cmp byte ptr finished flag, je main loop ; Se rulează în timp ce finished flag este zero mov bx OFFFFh ; Timer divider pentru frecventa , Hz caii reprogram pit ; Reprogramați cronometrul caii restore int ; Restaurați IRQO nojjlaster: ret buffer addr dw offset buffer ; Adresa octetului redat curent old int hdd ? ; Handler vechi INT , h (IRQO) finished flag db ; Steagul de finalizare ' nume de fișier db FILESPEC, ; Numele fișierului tada wav cu calea completă ; Handler INT h (IRQO) ; Trimite octeți din buffer către placa de sunet int h handler proc departe pusha; Salvați registre cmp byte ptr cs:finished flag, ; Dacă steagul este deja , je exit handler /; A nu face nimic mov di,word ptr cs: buffer addr ; În caz contrar: DI = adresa octetului curent mov'bl, Oh ; Comanda DSP b cai dsp write ; Ieșire imediată pe biți mov bl,byte ptr cs: [di] ; BL = octet de date către ieșire cai dsp write ■ - inc di ; DI = adresa următorului octet str ( di,offset buffer+ ; - lungime audio în tada wav Programare la nivel de port jb mov neterminat: mov exit handler: mută afară popa iret int h handler not finlshed byte ptr cuvânt ptr al, h h,al endp cs:finished flag, cs:buffer addr,di ; Manipulator de capăt; trimitere nespecifică; Restaurați registrele tamponul a fost trecut, flag setat la ; Eu cad ; instalare ; In caz contrar: ; salvați adresa curentă întrerupere hardware, EOI (vezi secțiunea ) DSP ; procedura dsp reset ; Resetare și inițializare dsp reset proc aproape mov dx,SBP RT+ ; Port h - resetarea registrului DSP mov al, Scrierea unuia la acesta începe inițializarea afară dx, al mov cx, ; O mică pauză dsploop: '■ în al, dx buclă dsploop mov al, ; Scrierea zero încheie inițializarea out ,dx,al ; DSP-ul este acum gata de funcționare ; Verificați dacă există un DSP adăugați dx, ; Port Eh - starea bufferului de citire DSP mov cx, checkport: în al dx; Citiți starea tamponului şi al, h; Verificați bitul jz port znot ready ; Dacă este zero, portul nu este încă pregătit sub dx, În caz contrar: portul Ah - citiți datele din DSP în al dx adăugați dx, ; Și din nou portul Eh cmp al OAAh ; Dacă se citește numărul AAh - DSP este prezent; și chiar gata de plecare • je good reset port nut ready: bucla check port ; Dacă nu, repetați testul de de ori bad reset: •stc ; si renunta ret; Ieșire cu CF = good reset: clc ; Dacă inițializarea a avut succes, ret; ieșire cu CF = dsp reset endp ; procedura dsp write ; Trimite un octet de la BL la DSP dsp write proc aproape mov dx,SBP RT+ Ch ; Port Ch - intrare de date/comandă DSP YANMVMMMMMMII Trucuri complexe de programare scrie bucla: ; Așteptați ca tamponul de scriere DSP să fie gata în al, dx; Citiți Portul Ch şi al, h; și verificați bitul jnz write loop ; Dacă nu este zero, mai așteptați mov al, bl ; In caz contrar: ' out dx al ; trimite date ret dsp write endp ; procedura reprogram pit ; Reprogramează canalul al temporizatorului de sistem la o nouă frecvență ; Intrare: VX = divizor de frecvență reprogram pit proc aproape cli ; Dezactivați întreruperile mov al, b; Canal , scrie octeți mici și înalți, ; modul de operare , format de contor - binar afară h,al ; Trimiteți acest lucru la primul registru de instrucțiuni de cronometru mov al, bl ; Divizor octet mic - afară h,al ; la canalul registrului de date mov al bh ; Și octet mare - afară h,al ; același fel sti; Acum IRQO este apelat cu o frecvență de O/BX Hz ret- reprogram pit endp ; procedura hook int ' ; Interceptează întreruperea INT h (IRQO) hook int proc aproape mov ax, h ; AH = h, AL = numărul de întreruperi int b; Obțineți adresa vechiului handler mov word ptr old int h,bx ; Stocați-l în old int h mov word ptr old int h+ ,es mov ax, h ' AH = h, AL = numărul de întreruperi mov dx,offset int h handler ; DS:DX - adresa handlerului int h I ; Instalați handler ret hook int endp procedura restore int Recuperează întreruperea INT h (IRQO) restore int proc pedg movax, h ; AH = h, AL = numărul de întreruperi Id-uri dx,dword ptr old int h ; DS: X - adresa handlerului int h; Instalați vechiul handler ret restore int endp ; procedura deschisa fişier ; Deschide fișierul filename și copiază datele de sunet din not (oh, considerându-l un fișier ; tada wav, a tampona* open file proc aproape mov ax, D h ; AH = Dh, AL = Programare la nivel de port mov dx, offset nume de fișier; DS:DX - nume de fișier ASCIZ cu calea int h; Deschideți fișierul pentru citire jc error exit ; Dacă fișierul nu poate fi deschis, ieșiți movbx,ax ; ID-ul fișierului în BX movax, h ; AH = h, AL = mov cx, ; CX:DX - valoare nouă de indicator mov dx, h; Datele din -tada wav încep la această adresă int h ; Mutați indicatorul fișierului mov ah, Fh ; AH = Fh mov cx, ; Aceasta este lungimea datelor audio din fișierul tada wav mov dx, buffer offset; DS:DX - adresa tampon int h; Citirea unui fișier ret eroare ieșire: ; Dacă fișierul nu a putut fi deschis mov ah, ; AH = h mov dx,offset notopenmsg ; DS:DX = mesaj de eroare int h; Deschideți fișierul pentru citire int h; Sfârșitul programului notopenmsg db „Eroare la deschiderea fișierului”,ODh,OAh,'$' open file endp tampon: ; Aici începe tamponul de de octeți sfârşitul începutului Dacă ați compilat programul latency asm din secțiunea și ați încercat să-l rulați în diferite condiții, este posibil să fi observat că sub Windows , precum și sub EMM și în unele alte situații, pauza dintre declanșarea efectivă a întreruperea temporizatorului și pornirea handler-ului pot fi foarte semnificative și variază în timp, astfel încât calitatea sunetului emis de programul nostru wavdir asm se va dovedi a fi foarte slabă sub EMM și într-o sarcină DOS sub Windows veți avea în general o respirație șuierătoare persistentă Pentru a evita acest lucru, precum și pentru a specifica rata exactă de biți a sunetului și a ieși audio pe biți, trebuie să treceți la programarea controlerului DMA (vezi sfârșitul secțiunii următoare pentru un exemplu de program care scoate audio folosind DMA ) Controler DMA Controlerul DMA este folosit pentru a face schimb de date între dispozitive externe și memorie Este necesar atunci când lucrați cu hard disk-uri și unități de disc, plăci de sunet și alte dispozitive care funcționează cu cantități semnificative de date Începând cu PC-ul AT, computerele au două patrullere DMA - biți (cu canalele , , și ) și biți (cu canalele , , și ) Canalul este folosit pentru a comunica cu unitățile de disc, canalul este folosit pentru hard disk, canalul se pierde atunci când controlerele sunt conectate în cascadă, iar scopul canalelor rămase poate varia DMA vă permite să citiți sau să scrieți un bloc de date începând de la o adresă liniară, care este descrisă ca un număr de de biți pentru primul controler DMA și ca un număr de de biți pentru al doilea, adică date pentru un - bit DMA trebuie să fie localizat în primul megaoctet de memorie, iar pentru al doilea - Trucuri complexe de programare în primii MB Cei patru biți superiori pentru adresele de de biți și cei biți superiori pentru adresele de de biți sunt stocați în registrele paginii DMA adresate prin porturile h - Fh: portul h: adresa paginii pentru canalul (biții - = biții - ai adresei) portul h: adresa paginii pentru canalul (biții - = biții - ai adresei) portul h: adresa paginii pentru canalul (biții - - biții - ai adresei) portul h: adresa paginii pentru canalul (biții - = biții - ai adresei) portul h: adresa paginii pentru canalul (biții - - biții - ai adresei) portul Ah: adresa de pagină pentru canalul (biții - = biții - ai adresei) • portul Bh: adresa de pagină pentru canalul (biții - = biții - din adresa) Adresa paginii determină începutul secțiunii de memorie de / kiloocteți cu care va funcționa acest canal, prin urmare, atunci când transferați date prin DMA, este imperativ să vă asigurați că nu există depășire a acestei secțiuni, adică nu există încercați să traversați adresa h: , h: , h: pentru primul DMA sau h: , h: , h: pentru al doilea Cei biți inferiori ai adresei sunt scrieți pe următoarele porturi: h: octeții - ai adresei blocului de date pentru canalul Olh: canalul octeți trimiși contor h - h: același lucru pentru canalul h - h: același lucru pentru canalul h - h: același lucru pentru canalul (două operații de citire/scriere sunt utilizate pentru aceste porturi - sunt transmiți mai întâi biții - , apoi biții - ) OCOh: biții - ai adresei blocului de date pentru canalul (bitul al adresei este întotdeauna zero) OClh: biții - ai adresei blocului de date pentru canalul C h: octet mic al contorului de cuvinte transmise canalului C h: octet mare al contorului de cuvinte transmise pe canalul C h - C h: același lucru pentru canalul C h - OCBh: același lucru pentru canalul OCCh - OCFh: același lucru pentru canalul (aceste porturi sunt concepute pentru a citi/scrie cuvinte întregi) Fiecare dintre aceste două controlere DMA are, de asemenea, propriul set de registre de control - registrele primului controler sunt adresate prin porturile h - OFh, iar al doilea - prin D - ODFh: portul h/ D h citiți: Registrul de stare DMA bit , , , : cererea DMA setată pe canalul / , / , / , / bit , , , : DMA sa încheiat pe canalul / , / , / , / portul de scriere h/D h: registru de comandă DMA (setat de BIOS) bit : semnalul DACK folosește un nivel ridicat , bit : semnalul DREQ folosește un nivel ridicat Programare la nivel de port bit : / : ciclu de scriere extins/întârziat bit : / : prioritățile se schimbă ciclic/fix bit : compresie în timp bit : controlerul DMA dezactivat bit : capturarea canalului activată (pentru modul memorie-la-memorie) bit : modul memorie-la-memorie activat (canal - canal ) portul h/ D h pentru scriere, registru de solicitare DMA bit : / : setați/ștergeți cererea pentru DMA biții - : numărul canalului ( , , , AND - / , / , / , / ) portul Ah/ D h pentru scriere: registru masca canalului DMA bit : / : setează/resetează bitul măștii biții - : numărul canalului ( , , , AND = / , / , / , / ) portul de scriere Bh/ D h: registru mod DMA biții - : - transmiterea cererii - transmisie unică (folosită pentru audio) - transfer în bloc (utilizat pentru discuri) - canalul este ocupat pentru cascadă bit : / : adresele scad/cresc bit : modul de inițializare automată biții - : - verifica - record - citire biți - : numărul canalului ( , , , AND - / , / , / , / ) portul de scriere Ch/ D h: comutați resetarea octetului scăzut/înalt Pentru citirea/scrierea valorilor de biți de la/la porturile de biți h - h Următorul octet trimis către aceste porturi va fi considerat octetul mic, iar următorul octet va fi octetul înalt portul de scriere Dh/ DAh: resetarea controlerului DMA Orice scriere aici are ca rezultat o resetare completă a controlerului DMA, deci trebuie reinițializat Port de citire ODh/ODAh: ultimul octet/cuvânt transferat portul Eh/ DCh pentru scriere: orice scriere elimină biții de mască de pe toate canalele portul Fh/ DEh pentru scriere: registrul masca tuturor canalelor: biții - : biții masca canalului / , / , / , / Cel mai adesea, dispozitivul extern însuși inițiază transferul de date și tot ce trebuie să facă programul este să scrie adresa de la începutul bufferului în porturile corespunzătoare canalului utilizat, lungimea blocului de date transmis minus unul în contor înregistrați canalul dorit, setați modul de funcționare al canalului și eliminați bitul de mască De exemplu, să ne întoarcem la programarea plăcilor de sunet și să schimbăm programul wavdir asm pentru a folosi DMA Trucuri complexe de programare wavdma asm Un exemplu de program care redă fișierul C:\WINDOWS\MEDIA\TADA WAV pe o placă de sunet folosind DMA FILESPEC equ "c:\windows\media\tada wav" ; Înlocuiți cu c:\windows\tada wav ; pentru versiunile mai vechi de Windows SBPORT equ h ; SBDMA equ ; procedura prdgram dma calculată ; doar pe canalul SBIRQ equ ; Doar IRQO - IRQ modelul minuscul cod org h ; Programul COM start: cai dsp reset ; Inițializare SP jc no blaster mov bl,OO h ; Comanda OD h cai dsp write ; Activați sunetul caii open file ; Citiți fișierul în buffer cal Г hook sblrq ; Prinde o întrerupere mov bl, h ; Echipa h caii " dsp write ; Setați viteza de transmisie mov bl, B h ; Constant pentru Hz/Stereo caii dsp write cai program dma ; Porniți transferul de date DMA bucla principala: ; Bucla principală ■ cmp byte ptr finished flag, je main loop ; Ieșiți când finished flag octet = caii restore sbirq ; Restabiliți întrerupere no blaster: ret old sbirqdd ? ; Adresa vechiului handler steag terminat db ; Steagul de terminare nume de fișier db FILESPEC, ; Nume de fișier ; Gestionarea întreruperilor plăcii de sunet ; Setează indicatorul finished flag la sbirq handler proc departe împinge toporul mov byte ptr cs:finished flag, ; Setați steag mov al, h; Trimiteți comanda EOI afară h,al ; la controlerul de întrerupere 'Toporul pop iret sbirq handler endp ; procedura dsp reset ; Resetare și inițializare DSP Programare activată dsp reset proc lângă \-* movdx,SBPORT+ ; Port h - resetarea registrului DSP • mov al, '-; Scrierea unuia începe inițializarea, afară dx,al mov cx, ; O mică pauză dsploop: în al, dx buclă dsploop mov al, ; Scrierea zero încheie inițializarea out dx,al ; DSP-ul este acum gata de funcționare adăugați dx, ; Port Eh - bitul când este citit indică ocupat mov cx, ; Buffer de scriere DSP checkport: în al,dx ; Citiți starea tamponului de scriere şi al, h; Dacă bitul este zero, jz port nu ready ; portul nu este încă gata sub dx, ; În caz contrar: portul Ah - citirea datelor din DSP în al dx adăugați dx, ; Portul este din nou Eh cmp al OAAh ; Verificați dacă DSP-ul returnează OAAh când citiți, - ; este un semnal de pregătire pentru muncă eu good reset- port nut ready: bucla check port ; Repetați testul pentru OAAh de de ori bad reset: stc ; Dacă Sound Blaster nu răspunde, ret; reveni cu CF = godd reset: clc ; Dacă inițializarea a avut succes, ret; reveni cu CF = dsp reset endp procedura dsp write ; Trimite un octet dsp write mov write loop: de la BL la DSP proc lângă dx,SBP RT+ Ch; Port Ch - intrare de date/comandă DSP ; Așteptați până când tamponul de scriere DSP este gata, in and jnz mov out ret dsp write al dx ; portul citit Ch al eoh - ; și verificați bitul write loop ; Dacă nu este zero, mai așteptați a , s; Altfel: dx,al ; trimite date endp ; procedura hook sbirq ; Interceptează întreruperea plăcii de sunet și o activează hook sbirq proc aproape mov ax, h+SBIRQ ; AH = h, AL = numărul de întreruperi int h; Obțineți adresa vechiului handler I LI NU Trucuri complexe de programare mov word ptr old sbirq,bx ; și salvează-l mov word ptr old sbirq+ ,es movax, h+SBIR ; AH = h, AL = numărul de întreruperi mov dx,offset sbirq handler ; Instalați un nou handler int h movcl, shl cl SBIRQ notcl; Construiți o mască de bit în al, h; Citiți OCW şi al, cl; Activați întrerupere afară h,al ; Scrieți CW ret hook sbirq endp procedura restore sbirq Restabilește handlerul și dezactivează întrerupere restore sbirq sproc aproape mov ax, O h+SBIRQ ; AH = h, AL = numărul de întreruperi Id-uri dx,dword ptr old sbi,rq int h; Restaurare handler movcl, shl cl SBIRO ; Construiți o mască de bit în 'al, h; Citiți OCW sau al, cl f; Dezactivați întrerupere afară h,al ; Notați QCW ret restore sbirq , endp h procedura open file ; Deschide numele fișierului și copiază datele de sunet din acesta, presupunând că ; acesta este tada wav, pentru a tampona open file proc peg mov ax, D h ; AH= Dh, AL= mov dx, offset nume de fișier; DS:DX - șir ASCIZ cu numele fișierului int h ; Deschideți fișierul pentru citire jc e rror exit ; Dacă fișierul nu poate fi deschis, ieșiți mov bx, ax ; ID-ul fișierului în BX movax, h ; AH = h, AL = ' mov cx, ; CX:DX - valoare nouă de indicator mov dx, h ; Datele încep de la această adresă; in tada wav int h; Mutați indicatorul fișierului mov ah, Fh ; AH = Fh mov cx, ; Aceasta este lungimea datelor din fișierul tada wav împinge ds ■ mov dx, ds și dx OFOOOh Aliniați tamponul la limita paginii K adăugați dx, OOOh pentru DMA mov ds, dx mov dx, eroare ieșire: • ; Dacă fișierul nu a putut fi deschis mov ah, ; AH = h mov dx,offset notopenmsg ; DS:DX = adresa mesajului de eroare int h'; Afișarea unui șir pe ecran int h; Sfârșitul programului ; mesaj de eroare notopenmsg db „Eroare la deschiderea fișierului”, ODh, OAh, „ $ ” open file endp ; procedura program dma ; Setează canalul DMA program dma proc aproape mov al, ; Masca canalul afară' O, al xor al, al; Resetează contorul afară Och, al mov al, h; Setați modul de transfer; (utilizați h pentru autoinițializare) afară OBh al împinge cs pop dx și dh OFOh adăugați dh, Oh; Calculați adresa tampon xor ax, ax afară h,al ; Scrieți biți mici afară h,al ; Scrieți următorii biți mov al,dh shr al, afară h,al ; Scrieți biți înalți mov ax, ; Lungimea datelor în tada wav dec ax ; OMA necesită lungimea minus unu afară h,al ; Scrieți cei biți inferiori ai lungimii mov al, ah afară h,al ; Scrieți cei biți superiori ai lungimii mov al, afară Oah, al ; Demasca canalul mov bl, h ; Echipa h cai dsp write ; Redare DMA simplă pe biți mov bx, ; Dimensiunea datelor d tada wav dec bx ; minus cai dsp write ; Scrieți pe DSP cu o lungime mai mică de biți mov bl, bh caii dsp write ■ ; si seniori ret / program dma endp sfârşitul începutului Trucuri complexe de programare În acest exemplu, se folosește modul normal de funcționare DMA, în care placa de sunet redă o secțiune de date, provoacă o întrerupere și, în timp ce gestionarul de întreruperi pregătește un nou buffer de date, programează DMA și placa de sunet pentru a continua redarea, trece ceva timp, care poate suna ca un clic Acest lucru poate fi evitat prin utilizarea modului de inițializare automată, care vă permite să faceți fără oprire în timpul redării Când utilizați modul DMA cu inițializare automată, trebuie să faceți următoarele: încărcați începutul sunetului redat într-un buffer cu o lungime de, de exemplu, KB și programați DMA să îl transmită cu inițializare automată Apoi spuneți DSP-ului că sunetul este redat cu auto-inițializare și dimensiunea bufferului este de KB În plus, atunci când vine o întrerupere de la placa de sunet, aceasta nu se va opri și va continua redarea din al doilea buffer de KB, deoarece este în modul de inițializare automată Acum să scriem următorul bloc de date în primii KB Când bufferul de K se epuizează, DMA-ul va începe să-l trimită din nou, pentru că l-am programat și pentru auto-inițializare (bit al portului OBh / OD h), DSP-ul va provoca o întrerupere și, de asemenea, nu se va opri, continuând redarea datele pe care le trimite controlerul DMA și, între timp, vom scrie următoarea secțiune a fișierului redat în al doilea KB din buffer etc Controler de întrerupere Un controler de întrerupere este un dispozitiv care primește cereri de întrerupere hardware de la toate dispozitivele externe Acesta stabilește ce cereri trebuie servite, care ar trebui să stea la coadă și care nu vor fi servite deloc Există două controlere de întrerupere, precum și DMA Primul controler, care deservește cereri de întrerupere de la IRQ la IRQ , este controlat prin porturile h și lh, iar al doilea (IRQ - IRQ ) prin porturile OAOh și OAlh Comenzile către controler sunt împărțite în comenzi de control (OCW) și inițializare (ICW): portul h/ A h pentru scriere- CW , OCW , ICW portul h/ A h pentru citire: vezi comanda CW portul h/ A h pentru citire și scriere: OCW - întreruperea mascarii portului h/ A h pentru scriere: ICW , ICW , ICW imediat după ICW Comenzi de control OCW : biții - : întrerupere - / - dezactivată Cu această comandă, puteți dezactiva sau activa temporar o anumită întrerupere hardware De exemplu, comenzi în al, lh sau al, b afară h,al duce la oprirea IRQ , adică a tastaturii Programare la nivel de port Am folosit CW în term asm pentru a activa întreruperea IRQ de la portul serial COM OCW : terminarea întreruperii și comenzile de schimbare a priorității „ biții - : comandă b: dezactivați schimbarea priorității în modul non-EOI b: EOI nespecific (sfârșitul întreruperii în modul prioritar) b: nicio operațiune b: EOI specific (sfârșitul întreruperii în modul fără prioritate) YOOB: activați schimbarea priorității în modul non-EOI b: Schimbare de prioritate cu EOI nespecific b: schimbare de prioritate b: schimbarea priorității cu biți specifici EOI - : b (indicați că acesta este OCW ) biții - : număr IRQ pentru comenzile b, IOb și b ' După cum sa menționat în Secțiunea , dacă au loc mai multe întreruperi în același timp, cea cu cea mai mare prioritate este deservită mai întâi La inițializarea controlerului, IRQ (întrerupere de la temporizatorul de sistem) are cea mai mare prioritate, iar IRQ are cea mai mică prioritate Toate întreruperile celui de-al doilea controler (IRQ - IRQ ) apar în această secvență între IRQ și IRQ , deoarece IRQ este folosit pentru a casca aceste două controlere Comenzile de schimbare a priorității vă permit să schimbați situația prin atribuirea întreruperii care se termină (comenzile sau ) sau în curs de procesare ( ) cu cea mai mică prioritate, următoarea întrerupere primind cea mai mare și așa mai departe Mai mult, în timp ce handlerul de întreruperi hardware se execută, nu există alte întreruperi cu priorități mai mici, chiar dacă handlerul a executat instrucțiunea sti Pentru a activa alte întreruperi, fiecare handler trebuie să trimită în mod necesar o comandă EOI - sfârşitul întreruperii - controlerului corespunzător De aceea, manipulatorii de întreruperi hardware din programele term asm și wavdma asm s-au încheiat cu comenzi mov al, h; comanda „sfârșit nespecific de întrerupere” afară h,al ; trimis la primul controler de întrerupere Dacă controlerul ar fi inițializat în modul non-priority, în loc de un EOI nespecific, ar trebui trimis unul specific care să conțină numărul de întrerupere în cei trei biți mici, dar BIOS-ul inițializează controlerul exact în modul prioritar În plus, controlerul ar putea fi inițializat în modul fără EOI, dar apoi în timpul funcționării gestionarului de întreruperi, ar putea apărea toate celelalte întreruperi, inclusiv cea procesată Modalitățile de inițializare a controlerului sunt discutate mai jos, dar aici vom lua în considerare ultima comandă de control OCW : Citiți starea controlerului și modul special de mascare bit : T"T~TTSH Trucuri complexe de programare biții - : mod special de mascare - nu schimbați - opriți - activare biții - : - indică că acesta este OCW bit : modul de sondare biții - : citiți starea controlerului - nu citi - citiți registrul cererilor de întrerupere - citiți registrul de întreruperi deservite În modul special de mascare, la executarea unui handler de întreruperi, toate întreruperile sunt activate, cu excepția celei care rulează în prezent și mascate de instrucțiunea OCW , ceea ce are sens dacă un handler de întreruperi cu o prioritate suficient de mare durează mult timp Cel mai adesea, OCW este folosit pentru a citi starea controlerului - cei doi biți inferiori selectează care dintre registrele controlerului vor fi returnate la citirea ulterioară din portul lh / Alh Ambele registre returnate au o structură similară cu QCW - fiecare bit corespunde IRQ-ului corespunzător Din registrul de cereri de întrerupere, puteți afla ce întreruperi au apărut, dar nu au fost încă procesate, și din registrul de întreruperi deservite, care întreruperi sunt procesate în acest moment Deci, o altă măsură de securitate pe care o folosesc programele rezidente este că nu puteți lucra cu unitatea de disc (IRQ ) dacă o întrerupere de la portul serial este în prezent deservită (IRQ ) și nu puteți lucra cu discul (IRQ / ) dacă întrerupere de la temporizatorul de sistem (IRQ ) Comenzi de inițializare Pentru a inițializa controlerul, BIOS-ul trimite o secvență de comenzi: ICW la portul h/ A h (diferă de OCW prin bitul său ) și ICW , ICW , ICW la portul lh/ Alh imediat după aceea IGW : biții - : b bit : / : declanșare la nivelul/marginea IRQ ( primit) bit : / : dimensiunea vectorului de întrerupere octeți/ octeți ( pentru x ) bit : fără cascadă, ICW nu va fi trimis bit : CW va fi trimis ICW : Numărul de gestionare a întreruperii pentru IRQ /IRQ (multiplu de opt) ( pentru primul controler, h pentru al doilea Unele sisteme de operare schimbă primul handler la h) ICW pentru controler principal: biții - : controler slave conectat la ieșirea - (O b în PC) ICW pentru controler slave: biții - : numărul de ieșire al controlerului master la care este conectat slave Programare la nivel de port ICW : biții - : O bit : controler în modul cu prioritate fixă biții - : mod: , - fără tampon - tamponat/slave - tamponat/lider bit : modul automat EOI (adică, handlerii nu trebuie să trimită EOI la controler) bit : - modul de compatibilitate Q ; - normal Repetând procedura de inițializare, programul poate, de exemplu, să modifice corespondența dintre manipulatorii de întreruperi și întreruperile hardware reale Mutând adresa de bază a primului controler într-o zonă nefolosită (de exemplu, h) și setând handlere personalizate pentru fiecare dintre întreruperile INT h - h care apelează INT h - OFh, puteți fi absolut sigur că niciun program nu va determina întreruperea hardware handler care a primit managementul mai devreme, al tău ; picnit asm ; Efectuează inițializarea completă a ambelor controlere de întrerupere ; cu maparea întreruperilor IRQO - IRQ la vectorii INT h - h, ; Programul rămâne rezident și emite un bip scurt după fiecare IR ; Recuperarea vechilor gestionari de întreruperi și reinițializarea ; controlerul la starea anterioară sunt omise model code org h minuscul ; Programul COM PIC BASE equ , h ; Procedura pic init va fi transferată la această adresă; IRQO - IR PIC BASE equ h ; Procedura 'pic init' va fi transferată la această adresă; IRQ - IRQ ' start: jmp end of resident ; Salt la începutul părții de instalare irqO handler: ; handler IRQO; (întreruperi de la temporizatorul sistemului) împinge toporul În al, h şi al, b; Opriți difuzorul afară h,al pop topor int h; Vechiul handler IRQO iret; A trimis EOI, așa că termină cu un simplu iret irqi handler: ; Handler IRQ (întrerupere de la tastatură) împinge toporul în al, h j | i ■NIIIII Trucuri complexe de programare sau al, b '; porniți difuzorul, afară h,al pop topor int h; Vechiul handler IRQ iret irq handler: ; Si asa mai departe int OAH iret irq handler: int OBh iret irq handler: int OC iret irq handler: int Odh iret irq handler: int OEh iret- irq handler: int „ofh iret sfârşitul rezidentului: ; Sfârșitul părții rezidențiale caii hook pic ints ; Instalarea handler-urilor noastre; INT h - h caii init pic ; Reinițializarea controlerului de întrerupere mov dx, offset end of resident int h; Lăsați noii noștri manageri rezidenți procedura init picl Efectuează inițializarea ambelor controlere de întrerupere, mapând IRQO - IRQ la PIC BASE - PIC BASE+ și IRQ - IRQ la PIC BASE - PIC BASE+ ; Pentru a reveni la starea standard, sunați ; PIC BASE = h, ; PIC BASE = h init pic proc aproape cli mov al, b; ICW afară h,al afară OAOh, al mov al,PIC BÂSE ; ICW LONG? primul controlor afară h,al mov al,PIC BASE ; ICW pentru al doilea controler afară A h,al mov al, h; ICW PENTRU primul controler afară h,al mov al, h; ICW PENTRU al doilea controler /■ Psy'ramirsmmine la nivel de port afară OAlh al mov al, b ouț h,al mov al, b afară A h,al stl ret init pic endp ; ICW pentru primul controler ; ICW PENTRU al doilea controler ; Interceptarea întreruperilor de la PIC BAȘE la PIC J ASE+ hook pic ints mov mov int mov mov int mov mov int mov mov int mov mov int mov mov int mov mov int mov mov int ret ret hook pic ints Sfârşit proc near ax, h+PIC BAȘE dx offset irqO handler h ax, h+PIC BASE dx offset irql handler h ax, h+PIC BA E dx offset irq handler h ax, h+PIC BASE dx offset irq handler h ax, h+PIC BASE dx offset irq handler h ax, h+PIC BASE dx offset irq handler h ax, h+PIC BASE dx offset irq handler h ax, h+PIC BASE dx offset irq handler h endp start Joystick Și în sfârșit - despre programarea joystick-ului Se conectează la încă unul, pe lângă portul serial și paralel, extern al computerului - la portul de jocuri Pentru portul de joc, spațiul portului I/O este rezervat de la h la Fh, dar la comunicarea cu joystick-ul se folosește un singur port - h, citirea din care returnează starea joystick-ului: portul h pentru citire: biții , : starea butoanelor , ale joystick-ului B biții , : starea butoanelor , ale joystick-ului A I î ii I Trucuri complexe de programare biții , : coordonatele y și x ale joystick-ului B biții , : coordonatele y și x ale joystick-ului A Cu starea butoanelor, totul este simplu - doar citiți un octet de la lh și determinați valoarea biților necesari Dar pentru a determina coordonatele joystick-ului, va trebui să efectuați o operație foarte incomodă și lentă: trebuie să scrieți orice număr în portul lh și să notați ora, citind constant starea joystick-ului Imediat după scrierea în port, biții de coordonate vor fi zero, iar timpul necesar pentru a se transforma la este proporțional cu coordonatele corespunzătoare (coordonatele X cresc de la stânga la dreapta, iar coordonatele Y - de sus în jos) Dacă joystick-ul nu este prezent, biții de coordonate fie vor fi unul de la bun început, fie vor rămâne zero pe termen nelimitat În plus, după scrierea în portul lh, aceeași acțiune nu poate fi efectuată din nou până când cel puțin unul dintre cei patru biți de coordonate se transformă în BIOS-ul are o funcție de întrerupere h h pentru lucrul cu un joystick, dar comunicarea directă cu porturile este mult mai rapidă și nu mult mai dificilă De exemplu, pentru a determina coordonatele joystick-ului, BIOS-ul efectuează până la patru cicluri de măsurare a coordonatelor, câte unul pentru fiecare Pentru a obține valoarea coordonatelor în unități rezonabile, determinăm cât de mult s-a schimbat contorul de canal al temporizatorului de sistem și împărțim acest număr la - rezultatul va fi același număr pe care îl returnează BIOS-ul Pentru un joystick standard ( kΩ) ar trebui să fie între - , deși de obicei valoarea maximă este în jur de Deoarece stick-urile analogice nu sunt dispozitive precise, coordonatele pentru aceeași poziție pot varia cu - , iar acest lucru este necesar luați în considerare mai ales la determinarea stării de odihnă, Să arătăm cum toate acestea pot fi implementate folosind exemplul citirii coordonatelor joystick-ului A: ; procedura read joystick ; Determină coordonatele curente ale joystick-ului A ; Ieșire: BP - coordonată Y, BX - coordonată X (- dacă joystick-ul ; nu răspunde), registrele nu sunt salvate read joystick proc aproape pushf ; Salvați steagurile cli ; și dezactivați întreruperile, deoarece ; se calculeaza timpul de executie a codului si nu ; trebuie să măsori și timpul de execuție; manipulatorii de întreruperi mov bx, - ; Valoarea lui X dacă joystick-ul nu răspunde movbp,bx ; Valoarea Y dacă joystick-ul nu „răspunde” mov dx, h ; Port mev al, afară h,al ; Blocați contorul de canal al temporizatorului în al, h Programare la nivel de port mov ah,al ' , ■ în al, h xchg ah, al; AX x valoarea contorului mov di,ax ; Notează-l la out dx,al Începeți să măsurați coordonatele joystick-ului în al dx; Citiți starea inițială a coordonatelor și al, b movcl,al ; Notează-l în CL read just ick loop: mov al, Out h,al ; Blocați contorul de canale temporizator în al, h mov ah, al in al, h' xchg ah,al ; AX ~ valoarea contorului , mov si,di ; SI - valoarea inițială a contorului sub si,ax ; SI - diferența de timp,' cmp zsi, FF h Dacă a avut loc un timeout (valoarea este luată din procedura BIOS), ja exit readj ; ieșire procedură în al,dx ; În caz contrar: citiți starea joystick-ului și al, b cmpal,cl ; compara-l cu precedentul eu read joystick loop xchg al cl ; Apăsați noua valoare la CL xor al cl ; și determinați bitul modificat test al, b; Dacă aceasta este coordonata x, ' Iz x la fel movbx,si ; scrieți coordonatele X în BX x same: test al, b; Dacă este o coordonată Y, jz read j oyst ic k loop mov bp si ; scrieți coordonatele Y în BP patru exit readj: testbx,bx ; Verificați dacă BX este egal cu - js bx bad shr bx, ; Dacă nu, împărțiți la bx bad: test bp bp ; Verificați dacă VR este - js bp bad shrbp, ; Dacă nu, împărțiți la bp bad: pop* ret readjoystick endp Dacă ați jucat vreodată cu un joystick, atunci probabil că știți procedura de calibrare atunci când jocul vă solicită să mutați joystick-ul peste două sau Trucuri complexe de programare patru colțuri Acest lucru trebuie făcut pentru a determina ce coordonate revine un anumit joystick pentru poziții extreme, deoarece chiar și pentru același joystick, aceste valori se pot schimba în timp Drivere de dispozitiv în DOS Așadar, în secțiunile anterioare, am vorbit despre modul în care unele dispozitive funcționează la cel mai scăzut nivel - nivelul porturilor I/O Cu toate acestea, programele de aplicație de obicei nu folosesc niciodată acest nivel, ci accesează toate dispozitivele prin instrumentele sistemului de operare DOS, la rândul său, se referă la instrumentele BIOS care comunică la nivel de port cu toate dispozitivele standard De fapt, procedurile BIOS îndeplinesc funcțiile driverelor de dispozitiv și ale programelor care oferă o interfață între sistemul de operare și hardware-ul computerului BIOS-ul știe de obicei cel mai bine cum să gestioneze dispozitivele care vin cu computerul, dar dacă doriți să conectați un dispozitiv nou despre care BIOS-ul nu știe, aveți nevoie de un driver de pornire special scris Driverele de dispozitiv în DOS sunt fișiere executabile cu o structură specială care sunt încărcate în etapa de pornire (când se execută comenzile DEVICE sau DEVICEHIGH din fișierul config sys) și devin de fapt parte a sistemului Driverul începe întotdeauna cu un antet de octeți: + : octeți - adresa îndepărtată a următorului driver DOS care urmează să fie încărcat - deoarece driverul va fi ultimul din lanț în momentul încărcării, adresa ar trebui să fie egală cu FFFFh: FFFFh + : octeți - atribute driver + : octeți - adresa procedurii strategiei + : octeți - adresa rutinei de întrerupere + Ah: octeți - numele driverului pentru dispozitivele cu caractere (completat cu spații) pentru dispozitivele bloc - octetul la offset OAh include numărul de dispozitive acceptate de acest driver, iar octeții rămași pot conține numele driverului Trebuie remarcat aici că DOS acceptă două tipuri de drivere - dispozitive de tip caracter și bloc Primul tip este utilizat pentru orice dispozitiv - tastatură, imprimantă, rețea, iar al doilea - numai pentru dispozitivele pe care pot exista sisteme de fișiere, adică pentru unități, discuri RAM, hard disk-uri non-standard, pentru a accesa partițiile de disc ocupate de alte sisteme de operare , etc Pentru a lucra cu un dispozitiv de caractere, programul trebuie să îl deschidă folosind funcția DOS „deschidere fișier) sau dispozitiv”, iar pentru a lucra cu un dispozitiv bloc, să acceseze discul logic corespunzător Deci, codul driverului dispozitivului este doar un cod de program obișnuit, la fel ca un fișier COM, dar nu trebuie să puneți directiva org h la început pentru a sări peste PSP De asemenea, este posibil să combinați driverul și programul executabil, Drivere de dispozitiv bDOS prin plasarea codului driverului în fișierul EXE cu un offset zero de la începutul segmentului și punctul de intrare al programului însuși mai jos La accesarea unui driver, DOS apelează mai întâi procedura de strategie (adresa la offset din antet), trecându-i adresa buffer-ului de solicitare care conține toți parametrii trecuți driverului, iar apoi procedura de întrerupere (adresa la offset ) fără orice parametri Procedura de strategie trebuie să stocheze adresa tamponului de solicitare, iar procedura de întrerupere trebuie să efectueze efectiv toate acțiunile necesare Structura buffer-ului de solicitare se modifică în funcție de tipul de comandă transmis driverului, dar structura antetului său rămâne aceeași: + h: octet - lungimea memoriei tampon de solicitare (inclusiv antetul) + h: octet - numărul dispozitivului (pentru dispozitive bloc) + h: octet - cod de comandă (OOh - h) + h: octeți - cuvântul de stare a driverului - trebuie completat de bitul driver : a apărut o eroare * biții - : THOOOO bit : dispozitiv ocupat bit : biți deserviți de comandă - : cod de eroare OOh: dispozitiv protejat la scriere Olh: dispozitiv necunoscut h: dispozitivul nu este pregătit s h: comandă necunoscută h: eroare CRC h: eroare tampon de solicitare h: eroare de căutare h: media necunoscută h: sectorul nu a fost găsit pe h: Nu mai există hârtie OAh: Eroare generală de scriere OBh: Eroare generală de citire AXIS: Eroare generală OFh: Schimbare neașteptată a discului + h: octeți - rezervat + Dh: zona de date începe de aici, diferită pentru diferite comenzi Chiar dacă driverul nu acceptă funcția cerută de la acesta, trebuie să seteze bitul al cuvântului de stare la Luați în considerare driverele de caracter și blocați cu exemple specifice Dispozitive de caractere Driverul de dispozitiv de caractere trebuie să conțină în câmpul atribute driver (offset în antet) unul în bitul cel mai semnificativ Apoi, biții rămași sunt tratați după cum urmează: J l I lJ ii Trucuri complexe de programare bit : bit : driverul acceptă funcțiile de citire/scriere IOCTL bit : driverul acceptă ieșirea înainte de funcția ocupată bit : bit : driverul acceptă funcțiile dispozitivului de deschidere/închidere biți - : bit : driverul acceptă funcția de solicitare a suportului IOCTL bit : driverul acceptă IOCTL generic bit : bit : driverul acceptă ieșire rapidă (prin INT h) bit : driver de dispozitiv de ceas bit : driver de dispozitiv NUL bit : driver de dispozitiv STDOUT , bit : driver de dispozitiv STDIN IOCTL este un set mare de funcții (peste cincizeci) disponibile ca diferite subfuncții ale INT h AH » h și concepute pentru a interacționa direct cu șoferii Dar despre IOCTL - puțin mai târziu, dar acum să ne familiarizăm cu cum funcționează, probabil, cel mai simplu dintre driverele cu adevărat utile Ca prim exemplu, luați în considerare un driver care nu servește deloc niciun dispozitiv, real sau virtual, ci pur și simplu mărește dimensiunea buffer-ului tastaturii BIOS la (sau mai multe) caractere Acest lucru ar putea fi realizat cu un program rezident normal, dar BIOS-ul stochează în zona sa de date doar cele mai apropiate adrese pentru acest buffer, adică un offset față de adresa de segment h Deoarece driverele sunt încărcate mai întâi în memorie, înainte de shell, ele se încadrează de obicei în intervalul de adrese liniare h - h, în timp ce acest lucru nu se poate întâmpla cu programele rezidente Driverul nostru va procesa o singură comandă, comanda de inițializare a driverului h Pentru ea, tamponul de solicitare arată astfel: + h: octet - h (lungimea tamponului de solicitare) + lh: octet - nu este utilizat + h: octet - (cod de comandă) + h: octet - cuvânt de stare a driverului (completat de driver) + h: octeți - neutilizat, + Dh: octet - numărul de dispozitive deservite (completat de driverul de bloc) + Eh: octeți - la intrare - sfârșitul memoriei disponibile pentru șofer; la ieșire - adresa primului octet din acea parte a driverului care nu va fi rezidentă (pentru a ieși fără instalare - trebuie să scrieți aici adresa primului octet) + h: octeți - la intrare - adresa șirului din CONFIG SYS care a încărcat driverul; la ieșire - adresa matricei VRV (pentru drivere de bloc) + h: octet - numărul primului disc + h: octeți - mesaj de eroare (OOOOh dacă nu a existat nicio eroare) - completat de șofer Drivere de dispozitiv în DOS zii Rutina de inițializare poate folosi funcțiile Olh DOS OCh, h, h și h ; kbdext asm ; Driver de dispozitiv de caractere care mărește tamponul tastaturii la BUF SIZE ; ( implicit) caractere BUFJJIZE equ , ' ; Noua dimensiune a tamponului modelul minuscul ; Pentru schimburi și apăsați h cod org ; Driverul începe cu CS:OOOO start: ; Titlul șoferului dd - ; Următoarea adresă a șoferului - OFFFFh:OFFFFh ; pentru ultimul dw h ; atribute; Dispozitiv de caractere, '; nu suporta nimic dw dw db offset strategie offset întrerupere „$$KBDEXT” Adresa procedurii strategiei Adresa rutinei de întrerupere Numele dispozitivului (Nu trebuie să fie același cu orice nume de fișier) cerere dd? ; Aici procedura de strategie stochează adresa tamponului de solicitare buffer db BUFJJIZE* dup (?); Și acesta este noul nostru tampon tastaturi cu o dimensiune de UF SIZE caractere (doi octeți per caracter!) procedura strategiei ; La intrare ES:BX = adresa tampon de solicitare strategie proc departe mov cs:word ptr request,bx ; Salvați această adresă pentru mov cs:word ptr request+ ,es ; proceduri de întrerupere ret strategie endp ; procedura de întrerupere întrerupe proc departe push ds ; Salvați registre „împingeți bx împinge toporul Ids bx, dword ptr cs:request ; DS:BX - adresa de solicitare mov ah,byte ptr [bx+ ] ; Citiți numărul comenzii sau ah, ah; Dacă comanda este ) (inițializare), jnz ■ieşire caii Ihit ; slujește-o ; In caz contrar: iesire: mov ax, h ; setați bitul (comandă servită) mov cuvânt ptr [bx+ ],ax ; în cuvântul de stare a șoferului POP ax > ; și restaurați ' + Ch: octeți - ignorat; , + Oh: octeți - EBX J ; TTL + h: bytes - EDX + h: bytes - ECX +lCh: bytes - EAX + h: bytes - FLAGS + h: bytes - ES + h: bytes - DS + h: bytes - FS + h: bytes - GS + Ah: bytes - IP + Ch: bytes - CS + Eh: bytes - SP Programare in PM' ■ + h: octeți - SS Valorile SS și SP pot fi zero, caz în care serverul DPMI însuși va oferi o stivă pentru ca întreruperea să funcționeze Pe lângă aceste funcții care transferă controlul către procedurile în modul real din modul protejat, există un mecanism care vă permite să faceți exact invers - transferați controlul către procedurile în modul protejat din real INT h, AX - t Alocați un punct de intrare pentru un apel din modul real Intrare: AH - OZOZZ DS:ESI = Selectorul procedurii modului protejat (se termină cu IRET) care urmează să fie apelat din real ES:EDI - locația selectorului structurii v regs care va fi folosită pentru a transfera registre Ieșire: dacă CF = , CX:DX = segment de decalaj punct de intrare Când controlul este transferat la o astfel de procedură, DS:ESI indică stiva de mod real, ES:EDI indică structura v regs, SS:ESP indică stiva furnizată de serverul DPMI, iar restul registrelor sunt nedefinite Numărul de puncte de intrare pe care le are un server DPMI este limitat și punctele de intrare neutilizate trebuie eliminate utilizând următoarea funcție DPMI INT h, AX „ h: Eliberați punctul de intrare pentru apelul din modul real Intrare: AX = h CX:DX - segment de decalaj punct de intrare Ieșire: CF = dacă punctul de intrare este eliminat Managerii de întrerupere Înainte de a ne uita la primul exemplu de program care utilizează DPMI, să ne concentrăm pe încă un grup de funcții ale acestuia - operațiuni cu manipulatori de întreruperi Când apare o întrerupere sau o excepție, controlul este transferat mai întâi prin lanțul de gestionare a întreruperilor în modul protejat, ultimul ■* ' handlerul - DPMI standard - intră în modul V , iar apoi controlul trece prin lanțul de manipulatori de întreruperi în modul real (manipulatorii de întreruperi și de excepții în modul real sunt aceiași) INT h, AX - h: Determinați adresa operatorului de întrerupere real Intrare: AX- h BL - numărul de întrerupere Ieșire: CF - întotdeauna , CX:DX - segment de gestionare a întreruperii în mod real INT h, AX - h: Setați manerul de întrerupere real Intrare: AX - h BL - numărul de întrerupere CX:DX „ Segment de gestionare a întreruperii în mod real Ieșire: CF - întotdeauna INT h, AX ~ h: Determinați adresa gestionarului de întrerupere protejat Intrare: AX - h BL „număr de întrerupere Ieșire: CF - întotdeauna , CX:EDX - selector de schimbare a operatorului INT h, AX - h: Setați manevrătorul de întreruperi protejat Intrare: AX- h BL - numărul de întrerupere CX:EDX = handler location selector Ieșire: CE- ' INT h, AX - h: Definiți adresa handler-ului de excepții Intrare: AX - h BL - numărul de excepție ( Oh - IFh) Ieșire: dacă CF este O, CX:EDX este selectorul de gestionare a excepțiilor INT h, AX " h: Setați gestionarea excepțiilor Intrare: AX- h BL” număr de excepție (OOh - IFh) CX:EDX = Selector de gestionare a excepțiilor Ieșire: CF - dacă nu au existat erori Dacă un handler de excepții transmite controlul mai jos în lanț către un handler standard de server DPMI, amintiți-vă că numai excepțiile , , , , , și sunt transmise manevrelor în mod real, iar excepțiile rămase determină programul termina Exemplu de program Acum să folosim interfața DPMI pentru a comuta în modul protejat și pentru a afișa un șir pe ecran Acest exemplu va funcționa numai pe sistemele care oferă DPMI pentru programele DOS care rulează pe acestea, cum ar fi Windows ooj i GT TI Programare în RM ; dpmiex asm ; Efectuează o comutare în modul protejat folosind DPMI ; La compilarea cu WASM, este necesară opțiunea /D WASM Modul protejat pe de biți a fost introdus în ; Segment de biți - se pregătește și comută în modul protejat RM seg segment octet utilizare publică presupune cs:RM seg, ds:PM seg, ss:RM stack ; Punct de intrare în program RM entry: ; Verificați DPMI mov ax, h ; Numărul h int Fh ; Întreruperea multiplexorului topor de încercare, topor; Dacă AX este diferit de zero, jnz DPMI error ; A apărut o eroare (DPMI lipsește) test bl, ; Dacă modul pe de biți nu este acceptat, jz DPMI error ; nici noi nu avem ce face Pregătiți adrese de bază pentru descriptorii viitori mov eax,PM seg mov ds, ax ; DS - adresa segment PM seg shl , -eax, ; EAX - adresa liniară a începutului segmentului PM seg mov dword Ptr PM seg addr, eax sau dword Ptr GDT flatCS+ ,eax ; Descriptor pentru CS sau dword Ptr GDT flatDȘ+ ,eax ; Descriptor PENTRU DS Stocați adresa procedurii de comutare DPMI mov word ptr DPMI ModeSwltch,di movword ptr DPMI ModeSwitch+ ,es ES trebuie să indice zona de date pentru comutarea modurilor Pentru noi, va coincide cu începutul viitoarei stive pe de biți adăugați eax, offset DPMI data; Adăugați offset la EAX shr eax, ; și transformați-o într-o adresă de segment inc ax se mişcă, ax ; Comutați în modul protejat > mov ax, ; AX = „G- vom fi a -a aplicație ifdef WASM db h; Corecție pentru wasm endif caii dword ptr DPMI ModeSwitch jcDPMI error ; Dacă comutarea nu a avut loc, ieșiți Acum suntem în modul protejat, dar limitele tuturor segmentelor sunt:- sunt setate la KB, iar lățimea de biți a segmentelor este setată la biți Trebuie să pregătim două selectoare - de biți cu o limită de GB - unul pentru cod și unul pentru date push ds ■ ' papi; ES a fost în general un segment PSP cu o limită de de ore interfata ORMI mov edi,offset GOT ; EDI este adresa tabelului GDT ; Buclă prin toți descriptorii din Navei GOT, mov edx, ; dintre care sunt doar două ( , ) sel loop: xor ax, ax ; Funcția DPMI : mov cx, int h; Creați un handle local mov word ptr selectorstedx* ],ax ; Salvare selector mov bx ax ; în tabelul selectorilor mov ax OOOCh ; Funcții DPMI OCh int h; Setați selectorul adaugă di, ; EDI este următorul descriptor dec dx jns sel loop / ; Încărcați selectorul de segment de cod în CS utilizând comanda RETF push dword ptr Sel flatCS ; Selector pentru CS ifdef WASM db h endif push offset PM entry ; EIR db h ; Prefixul mărimii operandului retf ■ Tranziție la de biți ; Controlul este transferat aici dacă a apărut o eroare la inițializarea DPMI ; (de obicei, dacă DPMI pur și simplu nu este acolo) DPMI error: împinge cs pop ds, mov dx, offset nodpmijnsg mișcare ah, h ; Afișarea unui șir pe ecran int h mov ah, Ch Sfârșitul programului EXE int h nodpmijnsg db „OPMI$ a eșuat” RM seg se termină ; Segmentul PM seg conține cod, date și stivă: pentru modul protejat RM seg segment octet utilizare publică presupune cs:PM seg,ds:PM seg,ss:PM seg' ; Tabel de descriptori GDT labei byte ; Descriptor pentru CS GDTflatCS db OFFh,OFFh,Oh,Oh,Oh,OFAh,OCFh,Oh ; Manevrează la OS GOT flatOS db OFFh,OFFh,Oh,Oh,Oh, F h,OCFh,Oh ; Punct de intrare în modul pe de biți - este încărcat numai CS PM entry: mov ax,word ptr Sel flatDS ; Selector pentru date mov ds,ax ; în D S segment ji I Programare în RM muta es ax ; în ES mov ss,ax i ; iar în SS mov esp, offset PM stack bottom ; Și instalați stiva •; De aici începe textul programului propriu-zis ; Programul funcționează în modelul de memorie plată cu o bază diferită de zero, ; baza lui CS, DS, ES și SS este aceeași și egală cu adresa liniară a începutului PM seg, ; toate limitele - GB mov ax, h ; Funcția DPMI h mov bx, h ; Întreruperea DOS h xorecx,ecx ; Nu copiați stiva , mov edi,offset v regs ; ES:EDI - adresa v regs int h; Apelați o întrerupere mov ah, Ch; Este singura cale int h; închideți corect programul DPMI hellojnsg db „Salut lumea din modul protejat pe de biți!$” v regs: ; Înregistrați valori pentru funcția DPMI h dd , , , , ; EDI, ESI, EUR, , EBX v edx dd offseț hellojnsg ; EDX dd ; ESC v eax dd h ; ЕАХ (AH = h, ieșire de linie pe ecran) dw ; STRAPURI, ES :EDI sunt la fel mov ax, h ; Funcția DPMI h - emulare mov bx,OO Oh ; întrerupe INT h org ecx ecx' int h; Obțineți informații despre mod Jc DPMI error cmp byte ptr v eax, Fh jne VBE error test octet ptr fs:[ ], h ; Bit al octetului de atribut = LFB da jz LFB error ; clădire ; Limită mov shl dec shr mov shr și descriptor de segment care descrie LFB eax ebp ; Memoria video în kiloocteți ex, ; Acum în octeți eax; Limita = dimensiune - ex, ; Limită în unități K, word ptr videodsc+ ,ax ; Scrieți biți - limită fiecare, ah OFh Programare în RM og byte ptr videodsc+ ,ah ; și biții - ai limitei mov eax ebp ; Memoria video în kiloocteți, shl eax, ; octeți mov edx,dword ptr fs:[ ] ; Adresa fizică a LFB mov cx,dx shld ebx,edx, ; Pune-l în CX:DX, mov di,ax shld esi,eax, ; iar dimensiunea este în : movax, h ; și apelați funcția DPMI h: int h; Hartă fizică adresă la liniar jc DPMI error shrd edx,ebx, ; Transferați adresa liniară primită mov dx, cx ; de la BS:CX la EDX mov word ptr videodsc+ , dx ; și scrieți biții - ai bazei, shr edx, mov byte ptr videodsc+ ,dl ; biții - mov byte ptr videodsc+ ,dh ; și biții - Drepturile mov bx, cs Iar cx,bx ; Citiți drepturile noastre de segment și cx, h ; și transferați biții DPL sau byte ptr videodsc+ ,ch ; la descriptorul în construcție ; Plasați descriptorul nostru în LDT și obțineți selectorul mov cx, ; Obțineți un mâner nou, movax, ; la DPMI int h jc DPMI error mov word ptr videosel ax ; Notează-i selectorul împinge ds ■ POP es mov edi, offset videodsc ; ES:EDI este adresa descriptorului nostru movbx,ax ; VX este selectorul oferit nouă mov ax OCh ; Funcția DPMI OCh: int h; Mânerul de încărcare la LDT jcDPMI error ; Acum in videosel exista un selector pe LFB ; Comutați la modul h ( h + LFB) mov word ptr v eax, F h ; F h - setați modul SVGA mov word ptr v ebx, h ; Mod h = h + LFB mov edi,offset v regs ; ES:EDI - structura v regs mov ax, h ; Funcția DPMI h mov bx, h ; Întreruperea emulării h xor ecx ecx int h mov ax,word ptr videosel ; AX este selectorul nostru enter flame: ; Aici vine controlul cu selectorul în AX la A h: , ; dacă apare o eroare în orice funcție WBE mov es,ax ; ES - selector de memorie video sau LFB Extensoare DOS Gі N! H De aici începe procedura de generare a flăcării Generarea paletei de flacără xor edi,edi ; Începeți să scrieți paleta la ES:OOOO horg ecx,ecx paleta gen: xor eax,eax ; Culorile încep de la , , movcl, ; Numărul de valori pentru o componentă paleta H': > stosb ; Scrieți un octet inc eax ; mări o componentă cmpsw ; sări peste doi octeți, loop palette H ; si asa de de ori push edi movcl, paleta : stosw; Scrieți doi octeți inc di ; și sări peste unul loop palette ; și așa de de ori (până la sfârșitul paletei) pop edit; Restaurați EDI inc di jns palette gen Paleta este generată, scrieți-o în registrele VGA DAC (vezi secțiunea ) mov al O ; Începeți de la registrul mov dx, C h ; Registrul index pentru scriere afară dx al împinge es push ds ; Schimbați ES și DS POP es POP ds xor esi,esi mov ecx, * ; Scrieți toate cele de registre mov edx, C h ; la portul de date VGA DAC rap outsb' împinge es push ds ; Schimbați ES și DS pop es pop ds ; Bucla principală este de a anima flacăra până când este apăsată orice tastă xor edx,edx ; Trebuie să fie zero mov ebp, h; Orice număr mov ecx,dword ptr scr width ; Lățimea ecranului bucla principala: împinge; Salvați es împinge ds papi; ES = DS - lucrăm doar cu buffer ; Animație cu flacără (algoritm clasic) inc ecx i Ț IMIMI ~ Programare în PM mov edit, offset buffer mov ebx,dword ptr scr height shr ebx, dec ebx mută esi,scr width animate flame: mov ax,[edi+esi* - ] ; Calculați valoarea medie a culorii adăugați al, ah; la un punct dat (EDI) din valori seteah; culori într-un punct din stânga și pe două linii mov dl,[edi+esi* + ] ; jos, dreapta și două linii în jos adăugați ax,dx mov dl,[edi+esi" ] ; și patru linii mai jos adăugați ax, dx ; cu prima valoare shr ax, ; modifica jz deja zero ; Reduceți luminozitatea culorii dec ax deja zero: stosb ; Scrieți noua culoare în tampon ■ adăugați eax,edx * shr eax, mov byte ptr[edi+esi- ],al buclă animate flame mov ecx,esi adăuga edi, ecx dec ebx jnz animate flame ; O bară pseudo-aleatorie în partea de jos a ecranului care servește drept generator de flăcări generator bar: xadd stosw stosw bp ax loop generator bar papi; Restaurați ES pentru afișare ; Ieșire de flacără către ecran xor edi,edi ; ES:EDI - LFB împinge esi adăugați esi, offset buffer; OS:ESI - buffer mov ecx,dword ptr scr size ; Dimensiunea tamponului în cuvinte duble rep movsd ; Copiați tamponul pe ecran pop esi mov ah, ; Dacă nu este apăsat int h; fară cheie, jz main loop ; continuați bucla principală mov ah, ; In caz contrar - int h; citește această cheie exit all: movax, h ; Restabiliți modul text Extensoare DOS int loh movax, C h 'int h ; AH= Ch ; Ieșiți din program sub extensia DOS Diversi manipulatori de erori DPMI eroare: ; Eroare la executarea uneia dintre funcțiile DPMI mov edx,offset DPMI error msg muta ah, int h; Afișează un mesaj de eroare jmp scurt exit all ; si iesi VBE eroare: ; Nu este acceptat de VBE mov edx,offset VBE error msg muta ah, int h; Afișează un mesaj de eroare jmp scurt ,start with vga ; și folosește VGA LFB eroare: ; LFB nu este acceptat mov edx, offset LFB error msg mov ah, ; Afișează un mesaj de eroare int h start with yga: mov ah, ; Așteptați ca orice tastă să fie apăsată int h mdv ax, h Comutare la modul video h, int h; x x mov ax, ; Funcția DPMI h: mov bx,OAOOOh ; construi un descriptor real int h; segment mov dword ptr scr width, ; Setați opțiunile de mod mov dword ptr scr lifeight, mov dword ptr Scr size, * / jmp enter flame ; și du-te la flacără date ; Diverse mesaje de eroare VBE error msg db „Eroare VBE ”,ODh,OAh db „Modul VGA x va fi folosit”,ODh,OAh,' $ DPMI error msg db „Eroare DPMI$” LFB error msg db "LFB nu este disponibil",ODh,OAh db „Modul VGA x va fi folosit”,ODh,OAh, ' ; Opțiuni pentru modul video scr widthdd scr heightdd scr dimensiuni * / ; Structura utilizată de funcția DPMI h v regs labei byte v edi dd v esi dd v ebp dd v res dd v ebx dd Î i i î i > NI! Programare în RM v edx dd v ecx dd v eax dd v flags dw v es dw v ds dw v fs dw v gs dw v ip dw v cs , dw v sp dw v ss dw ; Descriptorul de segment corespunzător LFB videodsc dw ; Limită de biți - dw ; Biții - ai bazei db ; Biții - ai bazei db b ; Acces db b ; Biții - ai limită și alți biți db ; Biții - ai bazei -' Selector de segment” care descrie LFB videosel dw '-r'- ' ■ date?' ; tampon de ecran buffer db * dup(?) ; grămadă stiva h sfârşitul -Start Programarea cu extensii DOS este una dintre cele mai bune prize pentru aplicațiile care trebuie să ruleze în orice mediu, inclusiv versiuni mai vechi de DOS, în timp ce necesită încă modul pe de biți Mai recent, majoritatea jocurilor pe calculator, în special celebrele Doom și Quake, au fost lansate tocmai ca programe folosind extensii DOS Astăzi, datorită omniprezenței sistemelor de operare pentru PC care funcționează în modul protejat pe de biți, cerința de a funcționa în orice versiuni de DOS devine din ce în ce mai puțin relevantă și tot mai multe programe sunt lansate numai în versiunile pentru Windows sau NT, care este subiectul capitolului următor Capitolul Programarea pentru Windows /NT Chiar dacă Windows /NT pare a fi un sistem de operare mai complex decât DOS, programarea lor în asamblare este mult mai ușoară Pe de o parte, o aplicație Windows rulează în modul pe de biți (nu luăm în considerare Windows și versiunile mai vechi care rulează în modul pe biți), dar cu un model de memorie plată, astfel încât programatorul obține toate beneficiile menționate în capitolul anterior, iar pe de altă parte, nu mai trebuie să studiem în detaliu modul de programare a diferitelor dispozitive de calculator la un nivel scăzut În mediile reale de operare, aplicațiile folosesc doar apeluri de sistem, care depășesc aici (aproximativ pentru Windows și pentru Windows NT) Toate aplicațiile Windows folosesc un format de fișier executabil special - formatul PE (Portable Executable) Astfel de fișiere încep ca fișiere EXE obișnuite în stil vechi (numite și MZ după primele două caractere ale antetului) Dacă un astfel de fișier este rulat din DOS, se va executa și va da un mesaj de eroare (textul mesajului depinde de compilatorul utilizat), în timp ce Windows va observa că după antetul obișnuit al fișierului MZ există un antet PE și va rula aplicarea Acest lucru va însemna doar că vor fi necesare alte opțiuni de pe linia de comandă pentru a compila programele Primul program Ca prim exemplu, să vedem cât de ușor este să scrieți un program sub Windows care încarcă un alt program În DOS (vezi secțiunea ), a trebuit să schimbăm alocarea memoriei, să completăm blocul de date EPB special și abia apoi să apelăm DOS Aici, nu numai că întreaga procedură este redusă la un singur apel de funcție, dar se dovedește și că puteți descărca programe, documente, fișiere grafice și text și chiar și adrese de e-mail și de internet exact în același mod - tot ceea ce face o acțiune este înregistrată în registrul Windows care se efectuează la încercarea de deschidere ; winurl asm ; Un exemplu de program este pentru win ; Lansează browserul implicit la adresa specificată în șirul URL ; În mod similar, puteți rula orice program, document și orice fișier, ; pentru care este definită operaţiunea vultur includ shell inc includ kernel inc I О ! ! —Mii Programming pentru Windows /NT model plat Adresa URL const db „http://www lionking org/~cubbi/'', start: cod; Eticheta punctului de intrare trebuie să înceapă cu un caracter de subliniere xor ebx', ebx • push ebx ; Pentru fișierele executabile, metoda de afișare push ebx ; Director de lucru push ebx ; Linie de comanda push offset URL ; Nume fișier cu calea push ebx ; Operațiunea vultur sau prinț (dacă NULL - deschis) push ebx ; ID-ul ferestrei care va primi mesajele cai ShellExecute ; ShellExecute (NULL, NULL, uri, NULL, NULL, NULL) push ebx ; Cod de ieșire caii ExitProcess ; ExitProcess(O) end start Deci, în acest program, două funcții de sistem win sunt numite - ShellExecute (deschide un fișier) și ExitProcess (încheie procesul) Pentru a activa o funcție de sistem Windows, un program trebuie să împingă toți parametrii de la ultimul până la primul din stivă și să transmită controlul CALL-ului îndepărtat Aceste funcții în sine eliberează stiva (terminând cu RET N) și returnează rezultatul muncii în registrul EAX Această convenție de trecere a parametrilor se numește STDCALL Pe de o parte, acest lucru vă permite să apelați funcții cu un număr nefix de parametri, iar pe de altă parte, apelantul nu trebuie să-și facă griji cu privire la eliberarea stivei În plus, funcțiile Windows salvează valoarea registrelor EBP, ESI, EDI și EBX, pe care le-am folosit în exemplul nostru - am stocat în registrul EBX și am folosit o comandă PUSH EBX de octet în loc de PUSH de octeți Înainte de a putea compila winurl asm, trebuie să creăm fișierele kernel inc și she! inc, unde vom plasa directivele care descriu funcțiile sistemului de apelat ; kernel inc • ; Includeți fișierul cu definițiile funcției din kernel dll ifdef TASM ; includelib import lib ; Numele funcțiilor utilizate extrn ExitProcess:near altceva; includelib kernel ib ; Numele adevărate ale funcțiilor utilizate sunt extrn imp ExitProcess® :dword • ; Misiuni pentru a face codul mai lizibil , ExitProcess equ imp ExitProcess@ endif ; shell inc : Includeți fișierul cu definițiile funcției din shell dll Primul program ІІINIMNNIKBZ ifdef TASM includelib import ib eu ; Numele funcțiilor utilizate, extrn ShellExecuteA:near ; Sarcini pentru a face codul mai ușor de citit ShellExecute equ ShellExecuteA altfel includelib shell lib ; Numele adevărate ale funcțiilor utilizate, extrn imp ShellExecuteA@ :dword ; Sarcini pentru a face codul mai ușor de citit ShellExecute equ imp ShellExecuteA@ endif Numele tuturor funcțiilor sistemului win sunt modificate astfel încât numele funcției să fie precedat de un caracter de subliniere și urmat de un semn @ și de numărul de octeți ocupați de parametrii trecuți acestuia pe stivă: astfel ExitProcess devine ExitProcess@ Compilatorii din limbaje de nivel înalt se opresc adesea acolo și apelează funcții numite ExitProcess@ , dar în realitate apare o mică procedură stub care nu face nimic, ci doar transferă controlul către aceeași etichetă, dar cu ip - imp ExitProcess@ adăugată În toate exemplele noastre, vom apela direct imp ExitProcess@ Din păcate, TASM (mai precis TLINK ) folosește propriul mod de a apela funcțiile sistemului, care nu poate fi ocolit în acest fel, iar programele compilate cu acesta se dovedesc a fi puțin mai mari și în unele cazuri rulează mai lent^ Am separat descrierile funcțiilor pentru TASM include fișiere atunci când ajutorul directivelor de asamblare condiționată, care le va folosi dacă /D TASM este specificat pe linia de comandă a asamblatorului În plus, toate funcțiile care funcționează cu șiruri de caractere (cum ar fi ShellExecute, de exemplu) există în două versiuni Dacă șirul este tratat în sensul obișnuit ca un set de caractere ASCII, un A (ShellExecuteA) este atașat la numele funcției O altă versiune a funcției, folosind șiruri în format UNICODE (doi octeți pe caracter), se termină cu litera U Toate exemplele noastre vor folosi funcții ASCII normale, dar dacă trebuie să recompilați programe UNICODE, trebuie doar să schimbați A la U în fișierele includ Deci, acum că avem toate fișierele necesare, putem compila primul nostru program Windows Compilarea MASM: ml /c /coff /cp winurl asm link winurl obj /subsystem:windows (în continuare, se utilizează versiunea pe de biți a link exe) Compilarea TASM: tasm /t /ml /D TASM winurl asm tlink /tre /aa /c /x winurl obj i 'iii | Programare pentru Windows /NT Compilarea WASM: wasm winurl asm wlink file winurl obj ferm windows nt op c ! Compilarea va necesita, de asemenea, fișierele kernel ib și shel! ib în primul și al treilea caz și import ib în al doilea Aceste fișiere sunt incluse în distribuțiile oricăror instrumente de dezvoltare win de la companiile respective - Microsoft, Watcom (Sybase) și Borland (Inprise), deși pot fi oricând recreate din fișierele kemel dll și shell dll aflate în WINDOWS\ directorul SYSTEM Uneori, împreună cu distribuțiile diferitelor instrumente de dezvoltare pentru Windows, este furnizat și un fișier windows inc, în care este dat macro-ul Invoke sau comanda caii este înlocuită cu o macrocomandă, astfel încât la apelare, puteți trece o listă de argumente, dintre care mai întâi va fi numele funcției care va fi apelată, iar apoi, despărțiți prin virgule, toți parametrii Folosind aceste definiții macro, programul nostru ar arăta astfel: start: xor ebx, ebx Invocați ShellExecute, ebx, ebx, offset(JRL, ebx,\ ebx, ebx Invocați ExitProcess, ebx end start' Și acest text este compilat în exact același cod ca al nostru, dar este apelată funcția intermediară ExitProcess@ , și nu funcția ітр ExitProcess@ Utilizarea acestei forme de notație nu permite utilizarea anumitor trucuri eficiente, optimizări, pe care le vom da în exemplele noastre - punerea în avans a parametrilor pe stivă și apelarea funcției cu comanda JMP Și, în sfârșit, este posibil să nu aveți un fișier windows inc, așa că vom scrie manual push înainte de fiecare parametru Aplicații de consolă Programele executabile pentru Windows sunt împărțite în două tipuri principale - aplicații de consolă și grafice La pornirea unei aplicații de consolă, se deschide o fereastră de text, cu care programul poate comunica cu WriteConsole / ReadConsole și alte funcții (respectiv, la lansarea din altă aplicație de consolă, de exemplu, managerul de fișiere FAR, consola curentă este alocată programul și controlul nu este returnat la FAR până la încheierea programului) În consecință, aplicațiile grafice nu primesc console și trebuie să deschidă ferestre pentru a afișa ceva pe ecran Pentru a compila aplicații de consolă, vom folosi următoarele comenzi MASM: ml /c /coff /cp winurl asm ,, , , , , link winurl asm /subsystem:console Aplicații de consolă TASM: tas" /t /ml /D TASM wihurl asm tlink Dre /ar /s /x winurl opj : WASM: wasm winurl asm fișierul wlink winurl obj formular Windows nt runtime console op c Încercați să compilați programul winurl asm în acest fel pentru a vedea cum funcționează o aplicație de consolă diferit de una grafică Ca exemplu de aplicație de consolă completă, să scriem un program care enumeră toate resursele de rețea conectate (discuri și imprimante) folosind funcțiile de sistem WNetOpenEnum, WNetEmimResource și WNetCloseEnum netenum asm O aplicație de consolă win care enumerează resursele de rețea include def inc include kernel inc include mpr inc model flat const greetjnessage db 'Exemplu de program de consolă win ',ODh,OAh,ODh,OAh, error message db ODh,OAh,'Nu s-a putut obține numele de utilizator currerit',ODh, Âh, error message db ODh,OAh,'Nu s-a putut enumera', ODh, OAh, good exit msg db ODh,OAh,ODh,OAh ,'Terminare normală',ODh,OAh, enumjnsgl db ODh,OAh,'Local', enum msg db ' remote - ', date user name db „Lista resurselor conectate pentru utilizator” user buff db dup(?) ; Buffer pentru WNetGetUser user buff l dd $-user buff • ; Dimensiunea tamponului pentru WNetGetUser enum buf l dd ; Lungimea lui enum t>iif în octeți enum entries data? dd ; Numărul de resurse care se încadrează în el enum buf NTRESOURCE ; Buffer pentru WNetEnumResource dd dup(?) ; de octeți pentru șiruri mesaj l dd ? ; Variabilă pentru WriteConsole, enum handle dd ? ; Identificator pentru WNetEnumResource cod start: Obțineți id-ul bufferului de ieșire stdout din sistem apăsați STD OUTPUT HANDLE caii GetStdHandle ; Returnează identificatorul STDOUT în eax, mov ebx, eax ; și îl vom stoca în EBX Imprimați șirul mesaj de salut pe ecran mov esi,offset greetjnessage caii output string Legea w і І j ПМІІ Programare pentru Windows /NT ; Determinați numele utilizatorului căruia îi aparține procesul nostru mută esi, offset user buff push offset user buff l ; Adresa unei variabile cu lungimea tamponului push esi ; adresa tampon apăsați ; NUL caii WNetGetUser cmp eax,NO ERROR ; Dacă apare o eroare, jne error exit ; Ieşi din cont, mov esi,offset user name ; în caz contrar, afișați șirul pe ecran caii output string ; Începeți să enumerați resursele rețelei push offset enum handle ; Identificator pentru WNetEnumResource apăsați push RESOURCEUSAGE CONNECTABLE ; Toate resursele atașate push RESOURCETYPE ANY ; Resurse de orice tip push RESOURCE CONNECTED ; Doar acum afiliat caii WNetOpenEnum ; Începeți listarea cmp eax,N ERR R ; Dacă apare o eroare, jne error exit ; Ieşi din cont Ciclul resurselor bucla enumerare; push offset enum buf l ; Lungimea tamponului în octeți push offset enum buf ; adresa tampon push offset enum entries ; Numărul de resurse push dword ptr enum handle ; Identificator de la WNetOpenEnum caii WNetEnumResource cmp eax,ERROR NO HORE ITEMS ; Dacă se epuizează je end enumeration ; lista completa, cmp eax,N ERR R ; dacă apare o eroare, jne error exit ; ieșiți cu un mesaj de eroare Afișați informații despre resursă pe ecran mov esi,offset enum msg ; Prima parte a liniei - caii ' output string ; la consolă mov esi,dword ptr enum buf IpLocalName ; Numele dispozitivului local - Zcaii output string ; la consolă mov esi,offset enum msg ; A doua parte a liniei - caii output string ; la consolă ' mov ■ esi,dword ptr enum buf lpRemoteName ; Nume dispozitiv la distanță - caii output string ; același fel jmp scurt enumeration loop ; Continuați listarea Sfârșitul ciclului end enumeration: push caii dword ptr enum hapdle WNetCloseEnum ; Sfârșitul enumerației mov esi,offset good exit msg exit program: caii putput string ; Ieșiți un șir apăsați ; Cod de ieșire Aplicații de consolă caii ExitProcess ; Sfârșitul programului ; Iese după erori error exit : mov esi, offset errorljnessage jmp short exit program error exit : mutare esi, offset error message jmp short exit prograra • ; Procedura ouțput string h ; Afișează un șir pe ecran ; Intrare: esi - • adresa de linie ; ebx • este identificatorul pentru stdout sau un alt buffer de consolă output string proc lângă ' ; Determinați lungimea șirului cld xor eax, eax mov edi,esi repne scasb dec edi sub edi,esi ; Trimite-l la consolă apăsați push offset message l ; Câți octeți sunt imprimați pe consolă push edi ; Câți octeți ar trebui să fie scoși la consolă push esi ; Adresa șirului de ieșit către consolă push ebx ; Identificatorul tamponului de ieșire caii WriteConsole ; WriteConsole(hConsoleOutput, IpvBuffe, cchToWrite, IpcchWritten', IpvReserved) ret șir ieșire endp ehd start În fișierul kemel inc, adăugați liniile dintre ifdef TASM și else: extrn • extrn GetStdHandle:near WriteConsoleA:near WriteConsole equ WriteConsoleA și între else și endif: extrn irap GetStdHandle® extrn imp—WriteConsoleA GetStdHandle equ imp GetStdHandle@ WriteConsole equ imp WriteConsoleA@ În plus, trebuie să creați un fișier trglps: ; trg ips ; Includeți fișierul cu definițiile funcției din mpr dll ifdef TASM includelib import ib ; Numele funcțiilor utilizate extrn WNetGetUserA:near extrn WNetOpenEnumA:near II Programare pentru Windows /NT extrn WNetEnumResourceA:near extrn WNetCloseEnum:near Atribuții pentru a face codul mai ușor de citit WNetGetUser equ WNetGetUserA WNetOpenEnum equ WNetOpenEnumA WNetEnumResource equ WNetEnumReso'urceA altfel includelibmpr lib ; Numele adevărate ale funcțiilor utilizate extrn imp WNetGetUserA@ :dword extrn imp WNet penEnumA@ :dword extrn imP WNet EnumResou rceA@ : dwo rd extrn imp WNetCloseEnum@ :dword ; Sarcini Pentru a face codul mai ușor de citit WNetGetUser equ imp WNetGetUse rA@ WNetOpenEnum equ imp WNetOpenEnumA@ WNetEnumResource equ imp WNetEnumResou rceA@ WNetCloseEnum equ imp WNetCloseEnum@ endif De asemenea, vom avea nevoie de un fișier def inc, unde vom plasa definițiile constantelor și structurilor din diverse fișiere include pentru limbajul C Există un utilitar h inc care convertește aceste fișiere în întregime, dar ne interesează propriile noastre ' include fișier, în care vom adăuga definiții noi după cum este necesar def inc Fișier cu definiții ale constantelor și tipurilor pentru programele exemplu sub win De la winbase h STD OUTPUT HANDLE equ - ; De la winerror h FĂRĂ EROARE equ ERROR NO MORE ITEMS equ ; De la winnetwk h RESOURCEUSAGE CONNECTABLE echivalent RESOURCETYPE ANYequ RESOURCE CONNECTED eqti NTRESOURCE struc dwScope, dd? dwTypedd ? dwDisplayType dd ? dwUs,agedd? IpLocalNamedd ? IpRemoteName dd ? IpCommentdd ? IpProvider dd ? NTRESOURCE se încheie Acest exemplu, desigur, ar fi putut fi construit mai eficient prin alocarea unui buffer mare pentru WNetEnumResource, cum ar fi cu LocalAPos sau GlobalAlloc (în win , acesta este același lucru), iar apoi, după ce ați citit informații despre toate resursele stocate în el, ar trebui să monitorizați dacă s-au epuizat sau nu și să apelați din nou WNetEnumResource Aplicații grafice Fereastra MessageBox Pentru a afișa orice fereastră, programul trebuie de obicei să-și descrie mai întâi aspectul și toate proprietățile, adică ceea ce se numește clasa ferestrei Vom vorbi despre cum să facem acest lucru puțin mai târziu, dar mai întâi vom afișa una dintre ferestrele cu o fereastră de clasă * predefinită de tip MessageBox MessageBox este o fereastră mică cu un mesaj text specificat și unul sau mai multe butoane În exemplul nostru, Mesajul va fi lumea tradițională Heno! ; winhello asm ; Aplicație grafică iip ; Afișează o fereastră + tip casetă de mesaj cu textul „Hello world!”, include def inc include kernel inc include user inc model plat const ; Titlul ferestrei hello title db „Primul program GUI win ”, ; Mesaj hellelsessage db „Bună lume!”, cod start: push MB ICONINFORMATION ; Puterea ferestrei push offset hello title ; Adresa liniei de titlu push offset hellojnessage ; Adresa liniei cu mesajul apăsați ; Un identificator de strămoș caii MessageBox împinge '; Cod de ieșire caii ExitProcess ; Sfârșitul programului end start Desigur, avem nevoie de noi completări la fișierele incluse: adăugați linia în fișierul def inc ; De la winuser fi MB ICONINFORMATION equ P și creați un fișier nou, user inc, care va include definițiile funcțiilor din user dll - biblioteca în care se află toate funcțiile principale responsabile pentru interfața ferestrei: i II Programare pentru Windows /NT ; user inc ; Includeți fișierul cu definițiile funcției din user dll ifdef „TASH includelib import ib ; Numele funcțiilor utilizate extrn MessageBoxA:near ; Sarcini ' , MessageBox equ MessageBoxA altfel' includelib user ib ; Numele adevărate ale funcțiilor utilizate extrn imp MessageBoxA@ : dword ; Sarcini pentru a face codul mai ușor de citit MessageBox equ imp MessageBoxA@ endif Acum putem compila acest program în același mod în care am compilat wmurLasm și îl rulăm - o fereastră mică cu mesajul nostru va apărea pe ecran, care va dispărea după apăsarea butonului OK Dacă compilam winhello asm ca aplicație de consolă, nimic nu se va schimba, dar caseta de text cu numele programului va fi deschisă până când se va închide fereastra cu mesajul nostru Fereastră Acum că știm cât de simplu este să afișați o fereastră cu o clasă predefinită, să începem să ne creăm propria fereastră - procedura pe care se vor baza toate exemplele ulterioare și să ne familiarizăm cu conceptul de mesaj În DOS, principalele mijloace de transfer al controlului către programe în diferite situații sunt întreruperile În Windows, întreruperile sunt folosite pentru nevoile sistemului, iar pentru aplicații există un mecanism similar - mecanismul de evenimente Deci, apăsarea unei taste de pe tastatură, dacă această tastă nu este utilizată de Windows, generează un mesaj WM KEYDOWN sau WMKEYUP, care poate fi interceptat prin adăugarea propriului lanț de gestionare a evenimentelor folosind SetWindowHookEx Evenimentele sunt apoi convertite în mesaje, care sunt trimise la funcții - handlere de mesaje - și care pot fi citite din programul principal folosind apelurile GetMessage și PeekMessage Pentru început, trebuie doar să procesăm mesajul de închidere a ferestrei (WM DESTROY și WM QUIT), prin care programul se va încheia ; fereastra asm ; O aplicație grafică iip care demonstrează rezultatul de bază al unei ferestre; include def inc include kernel inc: include user inc model, plat • date Aplicații grafice class name db "clasa ferestrei ", window name db "exemplu de asamblare win ", ; Structura care descrie clasa ferestrei, wc WNDCLASSEX ; Iată următoarele câmpuri: ; wc cbSize = * - dimensiunea acestei structuri ; wc style - stilul ferestrei (redesenați la redimensionare) ; wc lpfnWndProc - handler de evenimente ferestre (win proc) ; wc cbClsExtra *- numărul de octeți suplimentari după structură ( ) ; wc cbWndExtra - numărul de octeți suplimentari după fereastră ( ) ; wc hlpstance - ID-ul nostru de proces (?) ; wc hlcon - identificatorul pictogramei (?) ; wc hCursor - ID curbbra (?) ; wc hbrBackground - id-ul pensulei sau culoarea de fundal + (C L R WIND W+ ) ; wc IpszMenuName - resursă cu meniul principal ( în acest exemplu) ; wc lpszClassName - numele clasei (șir class name> ; wc hlconSm - identificator de pictogramă mic (numai pentru Windows , ; ar trebui să fie pentru NT) date? msg MSG ;Va aceasta este structura în care ; mesaj după GetMessage cod start: xor ebx, ebx ; EBX va avea pentru comenzile push ; Determinați ID-ul programului nostru împinge ebx caii UetModuleHandle mov esi,eax ; și salvați-l în ESI ; Completați și înregistrați clasa mov dword ptr wc hlnstance eax ; Un identificator de strămoș ; Selectați o pictogramă push IDI APPLICATION ; Pictograma aplicației standard push ebx ; ID modul cu pictogramă caii Loadlcon ' mov wc hlcon,eax ; ID-ul pictogramei pentru clasa noastră Selectați forma cursorului apăsați IDC ARROW ; săgeată standard ' push ebx ; ID modul cu cursor caii LoadCursor mov wc hCursor,eax ; ID cursor pentru clasa noastră push offset w c caii RegisterClassEx ; Înregistrați o clasă Creați o fereastră mov ecx,CW USEDEFAULT; push ecx este de cinci ori mai scurt decât push N push ebx ; Adresa structurii CREATESTRUCT (aici NULL) push esi ; Procesați ID-ul de primit z' • mesaje din fereastră (adică ale noastre) push ebx ; Identificatorul meniului sau ferestrei copil III Programare pentru Windows /NT push ebx ; Un identificator de fereastră strămoș push ecx ; Înălțime (CW USEDEFAULT - implicit) push ecx ; Lățimea (implicit) push ecx ; Coordonată Y (implicit) push ecx ; Coordonată X (implicit) push WS OVERLAPPEDWINDOW ; Stilul ferestrei push offset nume ferestre ; Titlul ferestrei push offset class name; Orice clasă înregistrată push ebx ; Stilul suplimentar caii CreateWindowEx Creați o fereastră (eax este identificatorul ferestrei) ' push eax ; Un identificator pentru UpdateWindow push SW SHOWNORMAL ; Tipul de afișare pentru ShowWindow push eax * ; Identificator pentru ShowWindow ; Nu mai avem nevoie de ID-ul ferestrei caii ShowWindow ; Afișează fereastra caii UpdateWindow ; și trimite-i un mesaj WM PAINT ; Bucla principală este verificarea mesajelor din fereastră și ieșirea cu WM QUIT mov edi,offset msg ; push edi este de ori mai scurt decât push N message loop: push ebx ; Ultimul mesaj Apăsaţi ebx; Primul mesaj push ebx ; ID fereastră ( - oricare dintre ferestrele noastre) push edi ; Adresa structurii MSG caii GetMessage ; Primiți un mesaj dintr-o fereastră de așteptare - test eax,eax ; nu uitați să utilizați PeekMessage, dacă trebuie să faci ceva în acest ciclu Dacă se primește WM OUIT, jz exit msg loop ; Ieşi din cont push edi ; În caz contrar, convertiți mesaje precum caii TranslateMessage ; Mesaje de la WM KEYUP la WM CHAR push caii edi DispatchMessage ; și trimiteți-le la procedura ferestrei (altfel este doar jmp short message loop nu poate fi închis) ; Continuați ciclul exit msg loop: ; Ieșiți din program împinge ebx caii ExitProcess procedura win proc Apelat de fereastră de fiecare dată când primește un mesaj Aici va avea loc toată munca din program Procedura nu trebuie să modifice registrele EUR, EDI, ESI și EBX! win proc proc ; Deoarece primim parametri pe stivă, construiți un cadru stivă, apăsați ebp mov ebp esp ; O procedură de tip WindowProc este apelată cu următorii parametri: Aplicații grafice wp hWnd equ dword ptr [ebp+ h]; ID fereastră, wp uMsg equ dword ptr [ebp+OCh]; numărul mesajului, wp wParam equ dword ptr [ebp+ Oh]; primul parametru, wp lParam equ dword ptr [ebp+ h}; al doilea parametru Dacă primim un mesaj WM DESTROY (ceea ce înseamnă că fereastra a fost deja eliminată de pe ecran apăsând Alt-F sau butonul din colțul din dreapta sus), atunci vom trimite un mesaj WM QUIT programului principal „trop wp uMsg,WM DESTOY jne not wm destroy Apăsați ; Cod de ieșire caii PostOuitMessage / Trimite WM QUIT jmp scurt end wm check ; și ieși din procedură not wm destroy: ; Dacă primim un alt message, apelăm handler-ul implicit Іеаѵе , ; restaura ebp jmp DefWindowProc ; și apelați DefWindowProc cu parametrii noștri ; și adresa de retur de pe stivă end wm check: Ieawe ; recupera ebp ret ; și ne întoarcem singuri, ștergând teancul de parametri win proc endp end start Adăugări necesare la fișierul def inc: ; De la wlnuser h IDI APPLICATION equ ( WM DESTROY equ ' CS HREDRAW equ CS VREDRAW • equ CW USEDEFAULT equ h WS OVERLAPPEDWINDOW equ OCFOOOOh IDC-ARROW equ SW SHOWNORMAL equ CULOARE-FERASTRĂ equ WNDCLASSEX, st ruc cbSizedd ? styje dd? IpfnWndProc dd ■ ? cbClsExtra dd ? cbWndExtra dd ? hlstance dd? hlcon dd? hCursor dd ? hbrBackground dd ?* IpszMenuName dd ? IpszClassName dd ? hlconSmdd ? WNDCLASSEX se încheie MSG struc hwnd dd? % І і i ! i fl Programare pentru Windows /NT mesaj dd? wParamdd ? IParam dd ? timp dd? - 'pt MSG dd final? s Adăugări la fișierul user inc: între ifdef TASM și else: extrn DispatchMessageA:near extrn T ranslateMessage:near r extrn GetMessageA:near extrn LoadIconA:near extrn UpdateWindow:aproape extrn ShowWindow:aproape extrn CreateWindowExA:near extrn DefWindowProcA:near extrn PostOuitMessage:near extrn RegisterClassExA:near extrn LoadCursorA:near DispatchMessage equ DispatchMessageA GetMessage equ GetMessageA Loadlcon- equ LoadlconA CreateWindowEx equ CreateWindowExA DefWindowProc equ ' DefWindowProcA RegisterClassEx equ RegisterClassExA LoadCursor equ LoadCursorA și între else și endif: extrn imp DispatchMessageA@ :dword extrn imp T ranslateMessage@ :dwo rd extrn imp GetMessageA@ :dword extrn imp Load!conA@ :dword extrn imp UpdateWindowW: dwo rd extrn imp ShowWi ndow® :dwo rd extrn imp CreateWindowExA@ :dword extrn imp DefWindowProcA@ :dword extrn imp Post uitMessage@ :dword extrn imp RegisterClassExA@ :dword extrn imp LoadCu rso r A@ :dwo rd DispatchMessage equ ^ imp DispatchMessageA@ TranslateMessage equ imp T ranslateMessage@ GetMessage equ imp GetMessageA@ loadlcon equ imp LoadIconA® UpdateWindow equ imp UpdateWindow@ ShowWindow equ imp ShowWindow@ CreateWindowEx equ imp CreateWindowExA@ DefWindowProc equ imp DefWindowProcA@ PostOuitMessage equ imp PostQuitMessage@ RegisterClassEx equ imp RegisterClassExA@ LoadCursor equ imp Load£ursorA@ Aplicații grafice " și în fișierul kemel inc între ifdef TASM și else: GetModuleHandle ext^n equ GetModuleHandleA:nea r GetModuleHandleA și între else și endif: extrn imp GetModuleHandleA :dword GetModuleHandle equ imp GetModuleHandjleA@ La începutul secțiunii, se spunea că programarea sub Windows este simplă, dar, în același timp, textul unui program obișnuit pentru afișarea unei ferestre goale pe ecran ocupă deja mai mult spațiu decât, de exemplu, textul din un program pentru redarea unui fișier WAV din secțiunea Unde este simplitatea promisă? Deci, se dovedește că, după ce am scris windov / asm, am creat deja o parte semnificativă din toate programele ulterioare și, atunci când completăm acest text cu un dialog cu drepturi depline, se va dovedi că nu mai trebuie să scriem toate aceste construcții greoaie, este suficient doar să copiați anumite secțiuni ale textului Meniul Meniul este una dintre pietrele de temelie ale ideologiei Windows Meniurile similare între ele vă permit să utilizați programe complet nefamiliare fără a citi instrucțiuni și să aflați despre capacitățile acestora prin vizualizarea conținutului diferitelor elemente de meniu Să încercăm să adăugăm un meniu și programului nostru window asm Primul lucru pe care trebuie să-l obținem este meniul în sine Acesta, precum și pictogramele, dialogurile și alte informații (până la versiunea programului), sunt scrise în fișierele de resurse Un fișier de resurse are extensia * RC pentru un fișier text sau * RES pentru un fișier binar creat de un compilator de resurse personalizat (RC, BRCC sau WRC) Ambele fișiere de resurse pot fi editate cu programe speciale incluse în distribuțiile C/C++ sau cu alte instrumente de dezvoltare Windows, dar nu vom crea un meniu prea complex și nu vom scrie fișierul RC manual, astfel: // winntenu rc C Fișier de resurse pentru programul winmenu asm "definiți ZZZ TEST ffdeflne ZZZ OPEN ' ' „definiți ZZZ SAVE #define ZZZ EXIT ZZZ Meniu MENIU { POPUP „&Fișier” { ' , MENIU „Open”, ZZZ OPEN MENIU „&Save”, ZZZ SAVE ; SEPARATOR DE MENIU MENIU „E&ieșire”, ZZZ EXIT } MENIU „&Editare”, ZZZ TEST }' Pentru a adăuga acest fișier în program, trebuie să îl compilați și să specificați numele fișierului res pentru linker: Programare pentru Windows /NT MASM: ml /s /coff/cp winmenu asm re /r winmenu rc link winmenu obj wlnmenu res /subsystem:windows TASM: tasm /x /m /ml /D TASM winmenu asm brcc wlnmenu res tlink /tre /aa /c winmenu obj,,,, winmenu res WASM: wasm winmenu asm wrc /r /bt=nt winmenu rc' fișierul wlink winmenu obj res wlnmenu res formează Windows nt op c ; '* msg MSG code ■ * început: * ★ xor ebx, ebx ;* push ebx ; * caii GetModuleHandle ■ ; * muta esi,eax ;* mov dword ptr wc hlnstance,eax ;* aplicații pentru mesaje grafice push IDI APPLICATION * push ebx * * caii Loadlcon f mov wc hlcon,eax ; * apăsați IDC ARROW • * push ebx • * caii LoadCursor ■ * mov wc hCursor,dax push offset w c caii RegisterClassEx push offset -menujiame ; Numele meniului , push esi ; ID-ul nostru caii LoadMenu ; Încărcați meniul din resurse mov ecx CW USEDEFAULT ■ * push ebx" * push esi '* push eax ; Meniu copil sau ID fereastră push ebx * ★ ' împinge ecx ' ★ împinge ecx ♦ împinge ecx împinge ecx ' * * f apăsați WSJWERLAPPEDWINDOW '* push offset window name ■ * push offset class name ★ push ebx'* caii CreateWindowEx • * push eax • * apăsați SW SHOWNORMAL • * push eax • * caii ShowWindow ■ * caii UpdateWindow mov edit,offset msg ■ * mesaj buclă: ■ * push ebx'* push ebx ; * push ebx * * push edit » caii ■ GetMessage • * test eax,eax , ■ * jz exit msg loop z; * push edit ■ » caii TranslateMessage ■ * push edit * * caii OispatchMessage ■ * jmp scurt mesaj loop ** exit msg loop: * * push ebx * caii ExitProcess ■ * ■ £ Qj AP ii n: Programare pentru Windows /NT procedura win proc Apelat de fereastră de fiecare dată când primește un mesaj ; Aici va avea loc toată munca din program ; Procedura nu trebuie să modifice registrele EUR, EDI, ESI și EBX! win proc proc ■ * push ebp • * mov ebp esp J * wp hWnd equ dword ptr tebp+O h] ;* wp uMsg equ dword ptr [ebp+OCh] ;* wp wParam equ dword ptr [ebp+ h] ;* wp lParam equ dword ptr [ebp+ h] ;* cmp wp uMsg,WM DESTROY î ★ jne not wm destroy • * apăsați * * caii PostOuitMessage : * jmp scurt end wm check * * not wm destroy: cmp wp uMsg,WM COMMAND ; Dacă avem WM COMMANO - jne not wm command ; asta este din meniul nostru mov eax,wp wParam ; și wParam conține submesajul nostru jmp dword ptr menu handlers[eax* ] ; Tranziție indirectă ; (în modul pe de biți, puteți sări la orice registru) menu handlers dd offset menu test,offset menu open dd offset menu save,offset menu exit ; Managerii de evenimente de testare, eagle și salvare scot un MessageBox ; Managerul de ieșire iese din program meniu test: mov eax,offset testjnsg ; Mesaj pentru MessageBox jmp scurt showjnsg menu open: , mov eax,offset openjnsg ; Mesaj pentru MessageBox jmp scurt showjnsg menu save: mov eax,offset savejnsg ; Mesaj pentru MessageBox show msg: apăsați MB OK ; Stil pentru MessageBox push offset nume meniu; Titlu push eax ; Mesaj push wp hWnd ; Un identificator de fereastră strămoș caii MessageBox ; Apel de funcție jmp scurt end wm check ' ; Ieșiți din win proc menu exit: ; Dacă ați selectat EXIT, apăsați wp hWnd caii ' DestroyWindow ; distruge-ne fereastra end wm check: părăsi;" xor eax, eax; Returnează ca rezultat al procedurii, ret ;" Aplicatii grafice nu wm comandă: ; not wm command pentru a scăpa de jmp suplimentar Pleacă :* , jmp DefWindowProc ; * win proc endp ; * ' end start ; * Deci, din de linii ale programului, doar s-au dovedit a fi noi, iar programul în sine, din punctul de vedere al utilizatorului, a devenit mult mai complicat Așa arată programarea sub Windows în asamblare - se ia un program șablon scris o dată pentru totdeauna, se modifică resursele și se scriu handlere pentru diverse evenimente de meniu și dialog De fapt, toată programarea este concentrată în aceste proceduri de manipulare Adăugările la fișierele include din acest exemplu sunt, de asemenea, minore în comparație cu windowasm În user inc între ifdef TASM și else: extrn LoadMenuA:near LoadMenu extrn DestroyWindow:near equ LoadMenuA și între else și endif: LoadMenu DestroyWindow și în def inc: ; De la winuser h WM COMMAND MB OK extrn imp LoadMenuA@ :dword extrn imp DestroyWindow@ :dword equ imp LoadMenuA@ equ imp DestroyWindow@ equ h equ O" Dialoguri Programele grafice pentru Windows nu se limitează aproape niciodată la un singur meniu, deoarece nu vă permite să introduceți informații reale - selectați doar orice articol dintre cele oferite Desigur, în bucla după GetMessage sau PeekMessage, puteți procesa mișcarea mouse-ului și evenimentele de apăsare a tastei, iar acest lucru se face în programe interactive, de exemplu, în jocuri, dar dacă trebuie să introduceți text și să îl editați mai târziu, selectați un fișier pe disc etc , apoi Dialogurile sunt principalul mijloc de introducere a informațiilor în programele Windows Dialogul este descris, precum și meniul, în fișierul de resurse, dar dacă meniul este ușor de scris manual, atunci de dragul dialogurilor, cel mai probabil, va trebui să utilizați un editor care vine cu compilatorul dvs preferat , cu condiția, desigur, să nu știți exact pe ce coordonate se află fiecare control (element de dialog activ) // windlg rc ț // Fișier de resurse care descrie dialogul utilizat în programul windlg asm // Toate definițiile următoare pot fi înlocuite cu „include , dacă există [| Programare pentru Windows /NT; // Stiluri pentru dialoguri, "deține DS CENTER "deține DS MODALFRAME "deține DS DL K // Stiluri pentru ferestre, "deține "deține" deține "deține "deține // Stiluri pentru editor "deține" deține WS MINIMIZEBOX WS SYSMENU WS VIZIBIL WS SUPRAPUT WS CAPTION ES AUTOHSCROLL ES STÂNGA ZDLG MENU x L x L x L x L x L x L „deține // Identificatorii comenzilor de dialog, „deține IDC EDIT "de-fine IDC BUTTON "deține IDC EXIT // Identificatori de elemente de meniu "deține IDM GETTEXT "deține IDM CLEAS "deține IDM EXIT ZZZ Dialog DIALOG , , , // x, y, latime, inaltime STYLE DS CENTER | DS MODALFRAME | DS DLOOK | WS CAPTION | WS MINIMIZEBOX | WS SYSMENU | WS VIZIBIL LEGITARE „Asamblare Win MENIU ZDLG MENU ÎNCEPE | Exemplu de dialog WS OVERLAPPED" Sfârşit BUTON EDITTEXT // Titlu // Meniul // Începutul listei de controale IDC EDIT, , , , ,ES AUTOHSCROLL | ES LEFT „E&ieșire”,IDC EXIT, , , , ZDLG MENU MENIU ÎNCEPE // Meniul Sfârşit POPUP „Test” ÎNCEPE MENUITEM MENUITEM MENUITEM MENUITEM „Obține Text”, IDM GETTEXT „Text clar”, IDM CLEAR SEPARATOR „Ieșire”, IDM EXIT Sfârşit Să folosim un exemplu simplu pentru a arăta cum poate fi folosit un dialog fără măcar a înregistra o nouă clasă Pentru a face acest lucru, trebuie să creați un dialog cu comanda CreateDialog sau una dintre variantele acesteia, fără a configura nicio fereastră strămoșească Toate mesajele din dialog și ferestrele pe care le creează vor fi trimise la o procedură de gestionare de tip DialogProc, similară cu WindowProc Aplicații grafice INMYA ; windlg asm ; Aplicație grafică iip care demonstrează cum se lucrează cu dialogul: ; Identificatori de controale (elemente de dialog) IDC EDlt IDC BUTTON IDC EXIT equ equ equ ; ID-urile elementelor de meniu IDM GETTEXT IDM CLEAR UI EXIT equ equ equ include def inc include kernel inc include user inc model date nume dialog date? buffer code start: plat db "ZZZ Dialog", ; Numele dialogului în resurse db dup(?) ; Buffer pentru textul introdus xor ebx, ebx ; EBX va fi pentru comenzile push' ; (mai scurt de ori) ; Determinați identificatorul programului nostru push caii ebx GetModuleHandle Începeți un dialog push push push push push ebx ; Valoarea care trebuie transmisă ca parametru WM INITBIALOG offset,dialog proc ; Adresa unei proceduri de tip DialogProc ebx; ID-ul ferestrei strămoși ( - fără dialog), offset dialog name ; Adresa numelui dialogului în resurse eax; Identificatorul programului ale cărui resurse conţin ; există un dialog (ID-ul nostru în EAX) caii DialogBoxParam ; Ieșiți din program push caii ebx ExitProcess ; procedura dialog proc edefine DS MODALFRAME „definiți S DL K „definiți WS POPUP ffdefineți WS CAPTION, „definiți WS SYSMENU „definiți IDOK „definiți IDC STATIC „definiți IDI APPLICATION „definiți WS BORDER x L x •OxSOOOOOOL // Dialog standard „Despre” ' ID DESPRE DIALOGUL DETERIAT , , , STYLEOS MODALFRAME | DS DL K | WS POPUP | WS CAPTION | WS SYSMENU CAPTION „Despre Asmpad ” { ICONA IDI APPLICATI N,IDC STATIC, , , , CTEXT „Asmpad ”,IDC STATIC, , , , CTEXT „Prototip de editor în stil notepad pentru Windows scris în întregime în limbaj de asamblare",IDC STATIC, , , , ,WS BORDER BUTONUL DE DEFP „ K”, ID K, , , , } , Apoi, luați în considerare textul programului ; winpad asm ; Aplicație grafică ip - editor de text || Programare pentru Windows /NT , -— | [Sunt pe include def inc include user inc include kernel inc include comdlg inc ID MENU equ h ID ACCEL equ h ID AB UT equ h MAXSIZE equ ; Numele maxim de fișier MEMSIZE equ ; Dimensiunea maximă temporară a memoriei tampon în memorie EditID egal cu model plat const c w name db "Asmpad ", ; Acesta este atât numele clasei, cât și numele ferestrei principale edit class db "edit", ; Nume de clasă predefinit pentru editor changesjnsg db „Salvați modificările?”, filter string db "AU Files", ,' , ; Măști pentru Get'FileName 't db "Fișiere text", ,'* txt', , date ; Structura folosită de Get*FileName ■; ofn OPENFILENAME ; O structură care descrie clasa noastră principală wc WNOCLASSEX ; = dacă numele fișierului nu este definit (fișier nou) date? h edit window dd ? ; Identificatorul ferestrei editorului h acceldd ? ; Identificator al matricei de acceleratoare p memorydd ? ; Adresă tampon în memorie SizeReadWrite dd ? SHAD MSGo rec RECT o buffer db MAXSIZE dup(?) ; Nume de fișier window title db MAXSIZE dup(?), dup(?) cod■ start: caii GetCommandLine ; Obțineți linia noastră de comandă mov edi,eax mișcare, '' mov ecx,MAXSIZE repne scasb; Găsiți sfârșitul numelui programului nostru cmp byte ptr[edi], eu cmdline empty X mov esi,edi U mov edit, offset buffer Aplicații grafice eu H!' ! —J J j ger movsb mov flag untitled, cmdline empty: ; Pregătiți și înregistrați clasa horg ebx,ebx caii GetModuleHandle mov esi,eax mov wc hlnstance,eax mov ofn hlnstance,eax apăsați IDI APPLICATION împinge ebx caii Loadlcon mov wc hlcon eax împingeți IDC SĂGEATĂ împinge ebx caii LoadCursor mov wc hCursor eax push offset w c caii RegisterClassEx ; Creați o fereastră principală ; Definiți-ne id-ul ; si salveaza-l wc hlnstance ; sau IDIJCON dacă pictograma este în resursă, ; sau esi dacă pictograma este în resurse ; Cursor predefinit (săgeată) împinge ebx împinge esi împinge ebx împinge ebx împinge împinge apăsați CWJJSEDEFAULT împinge CW UȘEDEFAULT /push WS OVERPEPEDWINDOW push offset c w name push offset c w name apăsați WS EX CLОENTEDGE caii CreateWindowEx împinge eax împinge eax împinge SW SHOWNORMAL împinge eax caii ShowWindow caii UpdateWindow ; Inițializați acceleratoarele apăsați IO-ACCEL împinge esi caii LoadAccelerators mov h accel,eax ; Bucla de așteptare a mesajului POP esi mov edit, offset msg mesaj bucla: împinge ebx împinge ebx ; Pentru pop esi înainte de message loop ; ESI - identificatorul ferestrei principale ; EOI - o structură cu un Mesaj din ea I j II Programare pentru Windows /NT împinge ebx push edi caii ' GetMessage ; Primiți un mesaj test eax,eax ; Dacă este WM QUIT, jz exit msg loop ; ieși din buclă push edi împinge h accel push esi ; hWnd caii TranslateAccelerator ; Convertiți acceleratoarele în IDM* test eax,eax jnz message loop push edi caii TranslateMessage ; Convertiți mesajele din taste push edi caii DispatchMessage ; și trimite înapoi jmp mesaj buclă scurtă exit msg loop: push msg wParam caii ExitProcess ; Sfârșitul programului ; Proceduri Win proc ; Procedura nu trebuie să modifice registrele win proc proc peg ; Parametri (inclusiv push ebp) wp hWnd equ dword ptr [ebp+ h] wp uMsg equ dword ptr [ebp+OCh] wp wParam equ dword ptr [ebp+ h] -wp lParam equ dword ptr [ebp+ h] EBP, EDI, ESI și EBX! Inițializam cadrul stivei împinge ebp mov ebp, în special pusha horg ebx,ebx mută esi,wp hWnd mov eax,wp^uMsg Procesați mesajul primit cmp P Global Lock@ : dwo rd extrn imp GlobalFree@ :dword extrn imp CreateFileA@ :dword extrn imp ReadFile@ :dword extrn ioip WriteFile@ O:dword Istrlen equ imp lstrlen@ GetCommandLine equ ■ imp GetCommandLineA@ CloseHandle equ imp CloseHandle@ GlobalAlloc equ imp GlobalAlloc@ GlobalLock equ imp GlobalLock@ GlobalFree equ imp GlobalFree@ CreateFile equ imp reateFileA@ ReadFile , equ imp ReadFile@ WriteFile equ imp WriteFile@ Adăugări la fișierul user inc: extrn LoadAcceleratorsA:near extrn TranslateAccelerator:near extrn SendMessageA:near extrn SetWindowTextA:near extrn MoveWindow:aproape extrn GetClientRect:near extrn GlobalUnlock: aproape LoadAccelerators equ LoadAcceleratorsA SendMessage equ SendMessageA SetWindowText equ SetWindowTextA și între else și endif: extrn imp LoadAcceleratorsA© ': dwo rd extrn imp TranslateAccelerat-orei : dword extrn imp SendMessageA@ :dword extrn imp SetWindowTextA@ :dword extrn imp MoveWindow@ : dword • extrn • ii'P GetClientRect@ : dword extrni imp GlobalUnlock@ :dword LoadAccelerators • equ imp LoadAccelerato rsA@ T ranslateAccelerator equ imp TranslateAccelerator@ SendMessage equ inip SendMessageA@ SetWindowText equ imp SetWindowTextA@ MoveWindow equ imp MoveWindow@ GetClientRect equ - imp GetClientRect@ GlobalUnlock equ imp GlobalUnlock@ Biblioteci dinamice În plus, avem nevoie de un nou fișier de includere, comdlg inc, care descrie funcțiile asociate cu apelurile de dialog standard (selectarea unui nume de fișier, tipărirea unui document, alegerea unui font etc ): ; comdlg inc ; Includeți fișierul cu funcții din comdlg dll ifdef TASM includelib import lib extrn GetOpenFileNameA:near extrn GetSaveFileNameA:near GetOpenFileName equ GetOpenFileNameA GetSaveFileName equ GetSaveFileNameA altfel includelib comdlg ib ; Numele adevărate ale funcțiilor utilizate extrn imp Get penFileNameA@ :dword extrn imp GetSaveFileNameA@ :dword ; Misiuni pentru ușurință în utilizare GetOpenFileName equ imp Get penFileNameA@ GetSaveFileName equ imp GetSaveFileNameA(Ș> endif Desigur, acest program poate fi îmbunătățit pentru o perioadă foarte lungă de timp - adăugați o bară de instrumente și o bară de stare, scrieți documentație sau faceți astfel încât să nu fie alocată o cantitate mică de memorie fixă pentru a transfera fișierul în editor, ci egală cu lungimea acestuia De asemenea, este permisă utilizarea funcțiilor de mapare a unei părți a unui fișier în memorie (CreateFileMapping, OpenFileMapping, MapViewOfFile, UnmapViewOfFile), care vă va permite să lucrați cu fișiere mari Există atât de multe funcții în API-ul Win încât poate dura foarte mult timp doar pentru a le studia Biblioteci dinamice Pe lângă aplicațiile obișnuite, Windows are un tip special de fișier numit biblioteci dinamice (DLL) Un DLL este un fișier care conține proceduri și date care sunt disponibile pentru programele care îl accesează De exemplu, toate funcțiile sistemului Windows pe care le-am folosit erau de fapt proceduri care făceau parte din astfel de biblioteci - kernel dll, user dll, comdlg dll etc Bibliotecile dinamice vă permit să reduceți utilizarea memoriei și dimensiunea fișierelor executabile în cazurile în care mai multe programe (sau chiar copii ale aceluiași program) folosesc aceeași procedură Putem considera că un DLL este un analog al unui program rezident pasiv, cu singura diferență că nu este în memorie dacă nu este încărcat niciun program care lucrează cu el Din punct de vedere al programării în asamblare, DLL este cel mai comun fișier executabil în format PE, diferind doar prin faptul că la introducerea acestuia sunt trecuți trei parametri pe stivă (identificatorul modulului DLL, motivul apelului) Programare pentru Windows /NT proceduri și un parametru rezervat) care trebuie eliminate, de exemplu, cu comanda ret Pe lângă această procedură, DLL include și altele, dintre care unele pot fi apelate din diferite programe Lista procedurilor de exportat trebuie să fie setată în momentul compilării DLL și, prin urmare, comenzile pentru crearea următorului nostru exemplu vor diferi de cele obișnuite Compilarea MASM: ml /s /cogg /cp /D MASM dllrus asm link dllrus obj ©dllrus lnk Conținutul fișierului dllrusdnk: /DLL /entry:start /subsistem:windows /export:koi win asm /export:koi win /export:koi wins asm ' /export:koi wins Compilarea TASM: tasm /m /x /ml /D TASM dllrus asm tlink -Tpd -c dllrus obj,,, dllrus def Conținutul fișierului dllrus def: EXPORTURI koi win asm koi win koi wins koi wins asm Compilarea WASM: wasm dllrus asm wlink ©dllrus dir Conținutul dllrus dir: fișier dllrus obj formular Windows nt DLL exp koi win asm,koi win,koi wins asm,koi wins dllrus asm DLL pentru win - transcoder de la koi la cp model plat ; Funcții definite într-un fișier DLL, ifdef MASM public Jkoi win asm@ public koi win@ public koi wins asm@ public koi wins@ koi win asm - Convertește un caracter în AL CHAR WINAPI koi win(simbol CHAR) koi wins asm - Convertește un șir în [EAX] VOID WINAPI koi win(CHAR* șir) altfel public public public public koi win asm koi win koi wins asm koi wins Aceleași funcții pentru TASM și WASM endif const Biblioteci dinamice II i i i i ; Tabel pentru traducerea unui caracter din codificarea KOI -G (RFC ) ; la codificarea Windows (cp ); tabel numai pentru caractere h FFh ; (adică scădeți h din caracter, convertiți-l cu xlat și adăugați din nou h) k w tbl db dup(O) ; Simboluri care nu există în cp , db dup(O); recodata la h db OOh, OOh, OOh, h, OOh, OOh, OOh, OOh db OOh, OOh, OOh, OOh, OOh, OOh, OOh, OOh db OOh, OOh, OOh, h, OOh, OOh, OOh, OOh db OOh, OOh, OOh, OOh, OOh, OOh, OOh, OOh db Eh, h, h, h, h, h, h, h db h, h, h, Ah, Bh, Ch, Dh, Eh db Fh, Fh, h, h, h, h, h, h db Ch, Bh, h, h, h, h, h, Ah db ' Eh, h, h, h, h, h, h, h db h, h, h, Ah, Bh, Ch, Dh, Eh db Fh, Fh, h, h, h, h, h, h db Ch, Bh, h, h, Dh, h, h, Ah cod OLLProcedura de intrare Obține trei parametri - un identificator, un motiv pentru apel și un parametru rezervat Nu avem nevoie de niciunul dintre ei start@ : mov ret al, ; Este necesar să returnați un număr diferit de zero în EAX ; Procedura BYTE WINAPI koi win (simbol BYTE) - punct de intrare pentru apelarea din C ifdef MASM ko! win® else koi win endif proc proc pop pop ecx ; Adresa de retur în ECX eax ; Parametru în ECX (stiva este acum șters; de parametri!) împinge ex ; Împingeți adresa de retur pe stiva pentru RET ; Nu există nicio comandă RET aici - controlul este transferat la următoarea procedură, ifdef MASM koi win@ else koi win endif endp endp Procedura koi win asm Punct de intrare pentru apelarea din programele de asamblare: introducere: AL - cod de caractere în KOI, ; ieșire: AL - codul aceluiași caracter în WIN ifdef MASM koi win asm® else koi win asm endif proc proc І І Programare pentru Windows test al, h; Dacă caracterul este mai mic de h (cel mai semnificativ bit ), jz dont decode ; nu recodificați push ebx ; in caz contrar - mov ebx, offset k w tbl subal, h; scade h, xlat; recod adăugați al, h ; si adauga h POP ebx dont decode: ret; Ieşi din cont ifdef MASM koi win asm@ endp altfel koi win asm endp endif ; VOID koi wins(BYTE*koistring) - ; punct de intrare pentru apelul de la C ifdef MASM koi wins@ proc altfel koi wins „proc endif pop ecx ; Adresa de retur din stivă POP eax ; Parametru în EAX push ecx ; Adresa de retur de pe stivă ifdef MASM koi wins@ endp altfel koi wins endp endif ; Punct de intrare pentru apelarea de la asamblator: ; intrare: EAX - adresa șirului care urmează să fie convertit din KOI în WIN, ifdef MASM koi wins asm@ proc altfel koi wins asm proc endif push esi ; Salvați registrele care ; nu poate fi schimbat push edi împinge ebx mov esi,eax ; Receptor cu șiruri mov edi,eax ; și potrivirea sursei mov ebx, offset k w tbl decode string: lodsb ; Citiți un octet test al, h; dacă bitul înalt este , /NT Biblioteci dinamice I si IN ; l" jz dont decode ; nu recodificați subal, h; altfel - scade h, transcodarea xlat adăugați al, h; si adauga h dont decode : stosb returnează un octet, test al, al; • dacă octetul nu este zero, jnz decode string '; continua pop ebx pop edit POP esi ret ifdef MASM koi wins asm@ endp altfel koi wins asm endp endif end start@ După cum puteți vedea din exemplu, a trebuit să denumim toate procedurile în mod diferit pentru diferiți asamblatori În cazul MASM, este clar că toate funcțiile trebuie să aibă nume precum start@ , altfel programele care le folosesc vor trebui să acceseze funcții cu nume precum imp start, adică un astfel de fișier DLL nu va fi încărcare dintr-un program scris în Microsoft C În cazul TASM și WASM, procedurile pot avea nume neschimbate (și mai mult, wlinkexe nu vă permite să exportați un nume de variabilă care conține simbolul @), deoarece compilatorii lor preiau nume de proceduri nu din un fișier de bibliotecă, dar direct dintr-un DLL folosind programul corespunzător - implib sau wlib Deci, pentru a utiliza fișierul DLL rezultat, să scriem un program simplu care va recodifica o linie de la KOI- g la Windows cp ; dlldemo asm ; O aplicație grafică pentru win care demonstrează cum se lucrează cu dllrus dll, ; imprimă șirul în K I și apoi în cp , recodat de funcția koi wins ; Compilare: ; MASM ; ml /c /coff /cp /D MASM dlldemo asm ; link dlldemo obj /subsystem:windows ; (fișierul dllrus lib creat la compilarea dllrus dll trebuie să fie prezent) ; TASM ; tasm /m /ml /D TASM dlldemo asm ; implib dllrus lib dllrus dll ; tlink /tre /aa /x /c dlldemo obj WASM wasm dlldemo asm j I Jl Programare pentru Windows /NT wlib dllrus lib dllrus dll fișierul wlink dlldemo obj din Windows nt include def inc include user inc includ kernel inc includelib dllrus lib ifndef MASM extrn koi win asm:near ; Definiții de funcții ;din DLL pentru extrn koi win:near ; TASM și WASM extrn koi wins asm:near ; (deși pentru WASM ar fi ;mai eficient extrn koi wins:near ; folosește imp koi win evidenţierea altceva; într-un bloc condiționat) extrn imp koi win asm© :dword ; Și asta este pentru MASM extrn imp koi win@ :dword extrn inip koi wins asm@ :dword extrn imp koi wins(& : dword koi win asm * equ imp koi win asm@ , koi win equ imp koi win@ koi wins asm equ i'np koi wins asm@ koi wins equ imp koi wins@ endif model plat const title stringl db „demo koi win: stririg în K I ”, title string db "koi win demo: șir în cp ", date koi string db F h, D h, D h,OCFh,OCBh,OC h, h,OCEh,OC h db h,OEBh,OEFh, E h, Dh, h, cod start: apăsați MB OK push offset title string ; Titlul ferestrei MessageBox push offset koi string ; Linie pe KOI apăsați caii MessageBox mov eax, offset koi string împinge eax caii koi wins apăsați MB OK push offset title string push offset koi string apăsați caii MessageBox Drivere de dispozitiv push Despre caii ExitProcess j II ijii Cod de ieșire Sfârșitul programului, sfârșitul start ' Acest mic fișier DLL poate fi foarte util pentru transcrierea textelor preluate de pe Internet sau din alte sisteme care folosesc codificarea KOI Folosind tabelele din Anexa , puteți extinde setul de funcții ale dllrus dll, până la recodificare la orice variantă Drivere de dispozitiv În Windows, precum și în DOS, există un alt tip de fișiere executabile - drivere de dispozitiv Windows xx și Windows utilizează un model de driver, Windows NT utilizează unul diferit, iar Windows utilizează un al treilea, deși în multe privințe similar cu modelul Windows NT Windows xx/ utilizează două tipuri de drivere de dispozitiv - drivere virtuale (VxD) care rulează cu nivelul de privilegii (de obicei pentru Windows xx și VXD pentru Windows ) și drivere neprivilegiate care rulează ca și programe obișnuite , cu nivel de privilegiu (de regulă, au extensia DRV) Windows NT utilizează un model de driver incompatibil numit kernel-mode Bazat pe modelul kernel-mode cu adăugarea de suport pentru tehnologia PNP și conceptul de fluxuri de date, modelul WDM (model de driver win ) a fost creat în , care este acum utilizat în Windows /NT și, aparent, va juca un rol major în viitor După cum v-ați putea aștepta, mijlocul principal de a crea drivere este asamblatorul și, în timp ce folosirea C este posibilă aici (spre deosebire de driverele DOS), este mai dificil: funcțiile apelate de driver pot trece parametri în registre; segmentele care alcătuiesc driverul trebuie denumite într-un anumit fel etc La crearea driverelor, de multe ori folosesc C și assembler în același timp, sau C și un pachet special care include toate instrumentele necesare inițializării Pentru a crea singur drivere pentru orice versiune de Windows, aveți nevoie de un set de programe, documentație, fișiere incluse și biblioteci distribuite de Microsoft, numit DDK - Drivers Development Kit (DDK-urile pentru Windows NT/ sunt gratuite ) Nu vom acoperi în detaliu programarea driverelor pentru Windows, deoarece există o mulțime de literatură dedicată acestui subiect, ca să nu mai vorbim de documentația care vine cu DDK Pentru a crea un driver, în orice caz, cel mai bine este să începeți cu unul dintre exemplele incluse și să schimbați/adăugați rutine de inițializare, gestionare de mesaje, întreruperi și excepții, handlere pentru API-ul furnizat de driver etc că este oarecum diferit din programele de asamblare cu care suntem obișnuiți Orice driver începe cu directiva include vmm inc, care include un fișier care conține definițiile segmentelor și macrocomenzilor utilizate M Sh Programare pentru Windows /NT Definițiile macro-urilor de forma VXD LOCKED CODE SEG/VXD LOCKED CODE ENDS corespund directivelor de început și de sfârșit a segmentului (în acest caz, segmentul LTEXT) Alte macrocomenzi importante sunt Declare Virtual Device și VMMCall/WDMCall Prima este pur și simplu o definiție care ia ca parametri ID-ul driverului, numele, versiunea, ordinea de încărcare și adresele rutinelor principale ale driverului din care este construit antetul acestuia Al doilea este un înlocuitor pentru comanda caii, care ia ca parametru numele funcției VMM sau WDM de apelat De exemplu, elementul de cod din driverul BIOSXLAT care prinde întreruperea h arată astfel: VxD ICOOE SEG ; Începutul segmentului ITEXT (segment de cod ; inițializare, execuție ; în modul protejat, care este eliminat ; din memorie după mesajul Init Complete) BeginProc BIOSXlat Sys Critical Init ; Procedura a cerut handler ; Mesaje Sys Critical Init - mai întâi; mesajul pe care îl primește șoferul ; De obicei, manipulatorii de mesaje ar trebui ; salvați registrele EBX, EDI, ESI și EUR, ; deși în acest caz este posibil ; a nu face mov esi,OFFSET BI SXlat Int ; Adresa handlerului INT h ; la registrul ESI Important de utilizat ; macro FFSET peste tot ; în loc de offset mov edx, t>h ; Orice număr care va ; se încadrează în EDX atunci când este sunat ; handler înregistrat VMMCall Alldcate PM Call Back ; Înregistrați punctul de intrare ; referindu-se la care, programe ; din modul protejat la VM va transfera controlul către procedura din driver (dar nu și programul ; ip ) - trebuie să folosească DevicelOControl pentru a lucra cu driverele jc BXLSCI NoPM ; Dacă CF = - a apărut o eroare: xchg edx, eax; punct de intrare - în EDX, numărul h - în EAX ; (acum acesta este numărul de întrerupere captabilă pentru Set PM Int Vector) mov ecx,edx shr ecx, h ; Selector punct de intrare movzx edx dx ; Decalaj punct de intrare VMMCall Set PM Int Vector ; Setați manerul de întrerupere INT h ; Dacă această funcție este apelată înainte de Sys VM Init, handlerul instalat ; devine o verigă în lanțul de handlere pentru toate mașinile virtuale ; După ce întreruperea trece prin lanțul de manipulatori ; în modul protejat, apare în V exact la fel ca în DPMI [cod pentru a intercepta alte întreruperi] EndProc BIOSXlat Sys Critical Init ; Sfârșitul procedurii VxD ICODE ENDS ; Sfârșitul segmentului de inițializare Drivere de dispozitiv i D În consecință, pentru ca această procedură să fie apelată pentru un mesaj Sys Critical Init, segmentul de cod fix LTEXT trebuie să conțină următoarea intrare: VxD LOCKED CODE SEG ; Începutul segmentului LTEXT WinProc BIOSXlat Control ; Începutul procedurii Control Dispatch Sys Critical Init, BIOSXlat Sys Critical Init ; Folosind o altă macrocomandă de la ѵvnp ips, înregistrați procedura BIOSXlat Sys Critical Init ca handler de mesaje Sys Critical Init CIC ; Procedura de gestionare a controlului ret; mesajele ar trebui să returneze CF = EndProc BIOSXlat Control ; Sfârșitul procedurii VxO LOCKEO CODE ENDS ; Sfârșitul segmentului LTEXT În cele din urmă, procedura BIOSXlat Control este înregistrată în antetul driverului ca o procedură care primește mesaje de control: ; Prima linie după p și include vmm inc: Declare Virtual Device BlOSXlat, , , BIOSXlat Control, BIOSXlat Device ID, BIOSXlat Init Order Nu este prea greu și, cu exemplele și documentația din DDK, precum și cu depanatorul SoftICE, puteți gestiona aproape orice sarcină care are sens să creați un driver Capitolul Asamblator și limbi de nivel înalt În capitolul anterior, în timp ce programam pentru Windows, am accesat deja proceduri scrise în limbaj de nivel înalt din programe de asamblare și, de asemenea, am creat proceduri de asamblare care pot fi accesate din limbaje de nivel înalt Pentru a face acest lucru, a fost necesar să se respecte anumite convenții despre trecerea parametrilor - parametrii au fost plasați pe stivă de la dreapta la stânga, rezultatul a fost returnat în EAX, stiva a fost eliberată de parametrii transferați prin procedura în sine Această convenție, cunoscută sub numele de STDCALL, nu este, desigur, singura, iar diferitele limbi de nivel înalt folosesc diferite moduri de transmitere a parametrilor Trecerea parametrilor Majoritatea limbilor de nivel înalt trec parametri la procedura apelată pe stivă și așteaptă ca parametrii să revină în registrul AX (EAX) Uneori se folosește DX:AX (EDX:EAX) dacă rezultatul nu se încadrează într-un registru și ST( ) dacă rezultatul este un număr în virgulă mobilă Convenția Pascal Cea mai evidentă modalitate de a exprima un apel către o procedură sau o funcție de limbaj de nivel înalt, după ce a decis că parametrii sunt trecuți pe stivă și returnați în registrul AX/EAX, este modul folosit în PASCAL (și, de asemenea, în BASIC, FORTRAN, ADA , OBERON, MODULA ) - doar împingeți parametrii de pe stivă în ordine naturală: record some proc(a, b, c, d, e) se transforma in: împinge un t împinge b push with push d push e caii some proc Aceasta înseamnă că procedura some proc, în primul rând, trebuie să șterge stiva la sfârșitul lucrului său (de exemplu, se termină cu comanda ret ) și, în al doilea rând, parametrii trecuți acesteia sunt pe stivă în ordine inversă: Trecerea parametrilor some proc proc bp bp sp ; Creați un cadru de stivă împinge mov a equ[bp+ ]; Definiții pentru acces ușor la parametri b equ [bp+ Q] c equ [bp+ ] ' d equ [bp+ ] e equ [bp+ ] [procedura text folosind parametrii a, b, c, d, e] pop bp ret some proc endp Acest cod corespunde exact formei complicate a directivei proc, care este acceptată de toți asamblatorii moderni: some proc proc PASCAL,a:word,b:word,c:word,d:word,e:word [text de procedură, cu parametrii a, b, c, d, e Deoarece BP este folosit ca indicator de cadru de stivă, nu poate fi folosit!] ret; Această comandă RET va fi înlocuită cu RET some proc endp Principalul dezavantaj al acestei abordări este dificultatea creării unei funcții cu un număr mutabil de parametri, similar cu funcția printf în limbajul C Pentru a determina numărul de parametri trecuți la printf, procedura trebuie să citească mai întâi primul parametru, dar nu nu-i cunosc locația pe stivă Această problemă este rezolvată prin abordarea folosită în C, în care parametrii sunt trecuți în ordine inversă Convenția C Această metodă de transmitere a parametrilor este utilizată în primul rând în C și C++, precum și în PROLOG etc Parametrii sunt împinși pe stivă în ordine inversă, iar eliminarea parametrilor din stivă (spre deosebire de convenția PAS C AL) este efectuate prin procedura de apelare: record some proc(a,b,c,d,e) se transforma in: împinge e împinge d împinge din împinge b împinge a caii some prqc ■ se adaugă sp, ; Eliberați stiva £ ^ i I I Vin ) Limbaje de asamblare și de nivel înalt Procedura apelată poate fi inițializată după cum urmează: unele proc proc împinge bp movbp,sp ; Creați un cadru de stivă şi equ [bp+ ]; Definiții pentru acces ușor la parametri b equ [bp+ ] cu equ [bp+ ] d equ [bp+ ] e equ [bp+ ] [textul procedurii folosind parametrii a, b, c, d, e] pop bp ret some proc endp Asamblatorii acceptă și acest format de apel prin „forma complicată a directivei proc cu limbajul C: some proc proc C, a:word, b:word, c:word, d:word, e:wo'r biții - : de biți cei mai importanți ai adresei fizice a tabelului de pointeri către fiecare jurnalele de pagină dacă bit PAE = bit ( +): bit PCD (Page Cache Disable) - acest bit împiedică încărcarea paginii curente în cache (de exemplu, dacă a avut loc o întrerupere și sistemul nu dorește ca handlerul de întrerupere să forțeze programul principal) din cache) bit ( +): PWT bit (Page Through Bit) - controlează modul în care paginile sunt scrise în memoria cache externă CR : Acest registru (a apărut doar la procesoarele Pentium) controlează noile caracteristici ale procesoarelor Toate aceste caracteristici sunt opționale și trebuie mai întâi verificate cu comanda CPUID bit : FSR - Activează comenzile FPU/MMX de salvare/restaurare rapidă FXSAVE și FXRSTOR (Pentium II) bit : RMC - permite executarea comenzii RDPMC pentru programe la toate nivelurile de privilegii (cu RMC - - doar la nivelul ) - Pentium Pro si superioare bit : PGE - Activează pagini globale (bit al atributului paginii) care nu sunt eliminate din TLB la comutatoarele de sarcini și scrie în CR (Pentium Pro și mai sus) bit : MCE - activați excepția #MC bit : PAE - include spațiu de adrese fizice de de biți - Pentium Pro și mai sus bit : PSE - activează modul de adresare cu pagini de MB bit : DE - dezactivează întreruperile de depanare la accesul la porturi bit : TSD r - dezactivează execuția comenzii RDTSC pentru toate programele, cu excepția programelor care rulează la nivelul de privilegii ojLi I UI Procesoare Intel în modul protejat bit : PVI - Permite semnalizarea VIF să ruleze în modul protejat, ceea ce poate permite unor programe scrise pentru nivelul de privilegii să ruleze la niveluri inferioare bit : VME - activați extensiile modului V - activați flag VIF pentru aplicațiile V Registre de depanare Aceste opt registre de de biți (DR - DR ) permit programelor care rulează la nivelul de privilegii să seteze puncte de întrerupere fără modificarea codului programului, cum ar fi pentru depanarea ROM sau programele care utilizează scheme complexe de protecție a urmei Un exemplu de depanator care utilizează aceste registre este SoftICE DR (DCR) - Registrul de control al remedierii biți - : câmp LEN pentru punctul de întrerupere (dimensiunea punctului de întrerupere) - octet - octeți - nedefinit (de exemplu, pentru a întrerupe la execuție) - octeți biți - : câmp R/W pentru punctul de întrerupere (tipul punctului de întrerupere) - la executarea unei comenzi -la înregistrare - la accesarea portului (dacă bitul DE din registrul CR = ) - când citesc sau scrii biții - : câmp LEN pentru punctul de întrerupere biții - : câmp R/W pentru punctul de întrerupere biții - : câmp LEN pentru punctul de întrerupere biții - : câmp R/W pentru punctul de întrerupere biții - : câmp LEN pentru punctul de întrerupere biții - : câmp R/W pentru punctul de întrerupere biți - : bit : GD bit - Activează un mod în care orice acces la registrul de depanare, chiar și din inelul de gardă , va arunca o excepție #DB (acest bit este șters automat în gestionarea excepțiilor) biții - : bit : bit GE - dacă acest bit este , punctul de întrerupere a accesului la date poate să nu lovească sau să lovească câteva instrucțiuni mai târziu, deci este mai bine să îl mențineți întotdeauna setat la bit : bit G - breakpoint activat bit : bit G - breakpoint activat bit : bit G - breakpoint activat bit : bit G - breakpoint activat biții , , , , : biții LE, L , L , LI, L - acționează la fel ca GE - G , dar se resetează la comutatorul sarcinii (puncte de întrerupere locale) Registrele DR (DSR) - registrul de stare de depanare - conține informații despre motivul întreruperii de depanare pentru Exception Handler #OB biții - : unități bit : Motivul întreruperii BT - bit de depanare în TSS al sarcinii la care tocmai a trecut ^um : BS - motivul întreruperii - semnalul de urmărire TF din registru FLAGS bit : BD - motivul întreruperii - următoarea comandă este pe cale să scrie sau să citească registrul de depanare, bitul GD din DR este setat la bit : Oh biții - : unii bit : OT - oprire la punctul finalizat bit : B - oprit la punctul bit : B - oprit la punctul bit : BO - oprit la punctul Procesorul nu șterge biții cauzați de întrerupere din acest registru, așa că handlerul de excepții #DB ar trebui să facă acest lucru singur În plus, o întrerupere poate apărea din mai multe motive în același timp, caz în care va fi setat mai mult de un bit DR - DR sunt rezervate Pe procesoarele pre-Pentium, sau dacă bitul DE al registrului CR este zero, accesarea acestor registre are ca rezultat accesarea DR și, respectiv, DR Dacă bitul DE este , apare o excepție #UD DR - DR conțin adrese liniare pe de biți a patru puncte posibile de acces la memorie Dacă sunt îndeplinite condițiile pentru o pauză de depanare, procesorul lansează o excepție #DB Registre specifice mașinii Acesta este un grup mare de registre (mai mult de o sută), al căror scop diferă în modelele de procesoare Intel și chiar uneori în procesoarele aceluiași model, dar de versiuni diferite De exemplu, registrele Pentium Pro MTRR ( de registre) descriu mecanismul de paginare care utilizează diferite zone ale memoriei - nu sunt stocate în cache, protejate la scriere, în cache transparent etc Registrele Pentium Pro MCG/MCI ( de registre) sunt utilizate pentru detectarea automată și procesarea erorilor hardware, registre Pentium TR ( registre) - pentru testarea cache-ului etc La descrierea comenzilor corespunzătoare, vom lua în considerare doar registrul Pentium TSC - contorul de cicluri al procesorului și un grup de patru registre Pentium Pro necesare numărării diverselor evenimente (număr de accesări la cache, înmulțiri, comenzi MMX etc ) Aceste registre s-au dovedit atât de utile încât au fost introduse comenzi suplimentare pentru a lucra cu ele - RDTSC și RDPMC Legea I j | | | | Procesoare Intel IU în modul protejat Sistem și comenzi privilegiate Procesor de comandă LGDT source Descărcați registrul GDTR Instrucțiunea încarcă valoarea sursă (o variabilă de octeți în memorie) în registrul GDTR Dacă operanzii actuali au o lățime de de biți, cei mai puțin semnificativi doi octeți ai operandului sunt utilizați ca dimensiune a tabelului de descriptor global, iar următorii patru octeți sunt utilizați ca adresă liniară Dacă operanzii actuali sunt de biți, numai octeții , , din operand sunt utilizați pentru adresa liniară, iar zerourile sunt scrise în octetul cel mai semnificativ al adresei Comanda este executată exclusiv în modul real sau cu CPL - Procesor de comandă Receptor SGDT Citiți registrul GDTR Plasează conținutul registrului GDTR în destinație (o variabilă de octeți în memorie) Dacă operanzii actuali au o lățime de biți, cel mai semnificativ octet al acestei variabile este umplut cu zero (începând cu și l-au umplut cu unii) Procesor de comandă Sursă LLDT ,-;J-, Registrul de încărcare LDTR Încarcă un registru LDTR pe baza selectorului din sursă (registru de biți sau variabilă) Dacă sursa este , toate comenzile, altele decât LAR, LSL, VERR și VERW, care se ocupă de acces de la LDT, vor genera o excepție #GP Comanda este executată numai în modul protejat cu CPL = Procesor de comandă Receptor SLDT Citiți registrul LDTR Plasează selectorul în registrul LDTR în destinație (registru sau variabilă de sau de biți) Acest selector indică un mâner din GDT al LDT-ului curent Dacă receptorul este pe de biți, cei biți superiori sunt setați la zero pe Pentium Pro și nu sunt definiți pe procesoarele anterioare Command rulează numai în modul protejat Procesor de comandă Sursa LTR Descărcați registrul TR Încarcă registrul de sarcini TR pe baza selectorului găsit în sursă (registru de biți sau variabilă) care indică segmentul de stare a activității (TSS) Această comandă este de obicei folosită pentru a încărca prima sarcină la inițializarea unui sistem multitasking Comanda este executată numai în modul protejat cu CPL = iii; ! Procesor de comandă Receptor STR Citiți registrul TR Plasează selectorul în registrul TR în destinație (registru sau variabilă de sau de biți) Acest selector indică un mâner din GDT care descrie TSS-ul sarcinii curente Dacă receptorul este pe de biți, cei biți superiori sunt setați la zero pe Pentium Pro și nu sunt definiți pe procesoarele anterioare Comanda este executată numai în modul protejat Procesor de comandă Registrul de încărcare sursă LIDT IDTR Încarcă valoarea sursă (o variabilă de octeți în memorie) în registrul IDTR Dacă operanzii actuali au o lățime de de biți, cei mai puțin semnificativi doi octeți ai operandului sunt utilizați ca dimensiune a tabelului de descriptor global, iar următorii patru octeți sunt utilizați ca adresă liniară Dacă operanzii actuali sunt de biți, numai octeții , , din operand sunt utilizați pentru adresa liniară, iar octetul cel mai semnificativ al adresei este setat la zero Comanda este executată exclusiv în modul real sau când CPL = Procesor de comandă Receptor SIDT Citiți registrul IDTR Plasează conținutul registrului GDTR în destinație (o variabilă de octeți în memorie) Dacă operanzii actuali au o lățime de biți, cel mai semnificativ octet al acestei variabile este umplut cu zero (începând cu și l-au umplut cu unii) Procesor de comandă Receptor MOV, sursă Redirecționarea datelor către/de la control/controlere și registre/e de depanare Receptorul sau sursa comenzii MOV pot fi registrele GR - CR și DR - DR În acest caz, celălalt operand al instrucțiunii trebuie să fie un registru de uz general pe de biți Când scrieți în registrul CR , toate intrările din TLB sunt resetate, cu excepția paginilor globale din Pentium Pro În timpul modificării bitului PE sau PG în CR și a bitului PGE, PSE sau PAE în CR , toate intrările din TLB sunt resetate fără excepție Comenzile sunt executate numai în modul real sau cu CPL = Procesor de comandă Sursa LMSW Încărcați cuvântul de stare a procesorului Copiază cei mai puțin semnificativi patru biți ai sursei (registru de biți sau variabilă) în CR , schimbând biții PE, MP, EM și TS De asemenea, dacă bitul PE este setat la , acesta nu poate fi resetat prin această comandă, adică nu puteți ieși din modul protejat jji ['I IUI Procesoare Intel în modul protejat Instrucțiunea LMSW există doar pentru compatibilitate cu procesorul și este întotdeauna mai convenabil să utilizați mov crO eax Comanda se execută numai în modul real sau cu CPL = Procesor de comandă Receptor SMSW Citiți cuvântul de stare procesor Copiază cei biți inferiori ai CR la destinație (registru de sau de biți sau variabilă de biți) Dacă receptorul este pe de biți, valorile celor mai semnificativi biți ai săi sunt nedefinite Instrucțiunea SMSW este necesară pentru compatibilitatea cu procesorul și este mai convenabil să utilizați în schimb mov eax, cr Procesor de comandă CLTS Clear TS flag în CR Comanda resetează bitul TS al registrului CR la , care este setat la de procesor după fiecare comutare de sarcină CLTS este conceput pentru a sincroniza salvarea / restabilirea stării FPU în sistemele de operare multitasking: prima comandă FPU într-o sarcină nouă Dacă TS = , va ridica o excepție #NM, al cărei handler va salva starea FPU pentru vechiul sarcină și restabiliți-o pe cea salvată anterior pentru cea nouă, după care va executa comanda CLTS și va reveni controlul Comanda este executată numai în modul real sau cu CPL - Procesor de comandă ARPL sink source Corectarea câmpului RPL al selectorului Comanda compară câmpurile RPL a două selectoare de segmente Destinația (registru de biți sau variabilă) îl conține pe primul, iar sursa (registru de biți) îl conține pe al doilea Dacă RPL receptor este mai mic decât RPL sursă, indicatorul ZF este setat și RPL receptor devine egal cu RPL sursă În caz contrar, ZF = și nu are loc nicio modificare De obicei, această comandă este folosită de sistemul de operare pentru a incrementa RPL-ul unui selector transmis de o aplicație pentru a se asigura că se potrivește cu nivelul de privilegii al aplicației (pe care sistemul îl poate prelua din RPL-ul segmentului de cod al aplicației de pe stivă) Comanda este executată numai în modul protejat (cu orice CPL) Procesor de comandă LAR sink source Permisiuni de citire a segmentului Copiază octeții de drepturi de acces de la descriptorul descris de selectorul situat în sursă (registru sau variabilă) în sursă (registru) și setează flag-ul ZE Dacă sunt utilizați operanzi pe biți, copiați doar octetul al descriptorului în octetul (biți - ) receptor Pentru de biți Comenzi de sistem Operanzii sunt copiați suplimentar Cei patru biți înalți (pentru segmentele de cod și date) sau întregul al șaselea octet al descriptorului (pentru segmentele de sistem) în octetul al receptorului Restul biților receptorului sunt setați la zero Dacă CPL > DPL sau RPL > DPL este pentru segmente de cod nesubordonate, dacă selectorul sau descriptorul este invalid sau în alte situații în care programul nu poate folosi acel selector, comanda LAR returnează ZF - Comanda este executată numai în modul protejat Procesor de comandă LSL destination source Citiți limita segmentului Copiază limita de segment (dimensiunea minus ) de la descriptorul al cărui selector se află în sursă (registru sau variabilă) în destinație (registru) și setează indicatorul ZF la Dacă bitul de granularitate din descriptor este setat și limita este stocată în unități de de octeți, comanda LSL își va converti valoarea în octeți Dacă sunt utilizați operanzi de biți și limita nu se potrivește în receptor, cei mai semnificativi biți ai săi se pierd Ca și în cazul LAR, această comandă verifică disponibilitatea segmentului din programul curent: dacă segmentul nu este disponibil, nimic nu este încărcat în receptor și indicatorul ZF este resetat la Comanda este executată numai în modul protejat Procesor de comandă Sursa VERR Sursa VERW Verificați permisiunile de citire Verificați permisiunile de scriere Comenzile verifică dacă codul sau segmentul de date al cărui selector se află în sursă (registru de biți sau variabilă) este disponibil pentru citire (VERR) sau scriere (VERW) de la nivelul de privilegiu curent Dacă segmentul este disponibil, atunci comenzile returnează ZF = , în caz contrar - ZF • Comenzile sunt executate numai în modul protejat Procesor de comandă INVD Resetează memoria cache WBINVD Scrieți și goliți memoria cache Aceste comenzi invalidează întregul conținut al cache-ului intern al procesorului și semnalează să golească memoria cache externă, așa că asta este tot după aceea accesările la memorie fac ca cache-ul să fie reumplut Comanda WBINVD salvează anterior conținutul cheii în memorie, comanda INVD provoacă pierderea tuturor informațiilor care au intrat în cache, dar nu au fost încă transferate în memorie Comenzile sunt executate numai în modul real sau cu CPL = Procesor de comandă Sursa INVLPG Void pagina Procesoare Intel în modul protejat Invalidează (invalidează) elementul TLB care descrie pagina de memorie care conține sursa (adresa în memorie) Comanda se execută numai în modul real sau cu CPL = Procesor de comandă HLT Opriți procesorul Pune procesorul într-o stare de oprire, din care doar o întrerupere hardware sau o repornire îl poate trezi Dacă cauza a fost o întrerupere, atunci adresa de retur împinsă pe stivă pentru handler-ul de întrerupere indică la următoarea instrucțiune după HLT Comanda se execută numai în modul real sau cu CPL = Procesor de comandă RSM Ieșiți din modul SMM P Este folosit pentru a scoate procesorul din modul SMM, care este folosit pentru a salva starea sistemului în situații critice (de exemplu, în timpul unei întreruperi de curent) La introducerea SMM (apare atunci când semnalul corespunzător este primit de procesor de pe placa de bază), toate registrele, inclusiv cele de sistem, și alte informații sunt stocate într-un bloc de memorie special - SMRAM, iar la ieșire (care este efectuată de RSM comanda), totul este restaurat Comanda este executată numai în modul SMM Procesor de comandă RDMSR Citiți din registrul MSR P WRMSR Scrieți în registrul MSR P Plasează conținutul registrului specific mașinii numerotat în ECX în perechea de registre EDX:EAX (RDMSR) sau conținutul registrelor EDX:EAX în registrul specific mașinii numerotat în ECX O încercare de a citi/scrie un MSR care este rezervat sau lipsește din acest model are ca rezultat o excepție #GP( ) Comanda se execută Doar în modul real sau cu CPL - Procesor de comandă RDTSC Citire de la contorul de ceas al procesorului P Plasează în perechea registrului EDX:EAX valoarea curentă a contorului de ceas al registrului TSC specific mașinii de - de biți, a cărui valoare a fost incrementată cu pentru fiecare ciclu de procesor de la ultima resetare Acest registru specific mașinii este disponibil pentru citire și scriere folosind comenzile RDMSR / WRMSR ca registru numărul Oh, iar pe Pentium Pro, când scrieți în el, cei de biți superiori sunt întotdeauna setați la zero Din moment ce registrele specifice maşinii Comenzi de sistem eu este posibil să nu fie prezent pe anumite modele de procesoare, prezența acestora ar trebui să fie întotdeauna determinată de CPUID (bit în EDX - prezența TSC) Comanda este executată la orice nivel de privilegii dacă bitul TSD din registrul CR este zero și numai în modul real sau cu CPE - dacă bitul TSD este Procesor de comandă RDPMC Citiți de la contorul de evenimente P Plasează valoarea unuia dintre cele două contoare de evenimente programabile (registrele de de biți specifice mașinii Clh și C h pentru Pentium Pro și Pentium II) în perechea de registre EDX:EAX Alegerea unui registru care poate fi citit este determinată de numărul sau din ECX Există registre similare pe Pentium (și Cyrіx x MX), dar au numere llh și h și pot fi accesate numai folosind comenzile RDMSR/WRMSR Modul de selectare a tipului de evenimente numărate diferă și în Pentium și Pentium Pro - pentru Pentium trebuie să scrieți în registrul MSR lh pe de biți, dintre care diferite cuvinte duble controlează alegerea modului fiecărui contor și tipul de eveniment, iar pentru Pentium Pro / Pentium II trebuie să scrieți scris pentru a înregistra b pentru contorul și b pentru contorul În consecință, seturile de evenimente dintre aceste procesoare diferă foarte mult: de evenimente pe Pentium, - pe Pentium Pro și - pe Pentium II i Procesor de comandă SYSENTER Apel de sistem rapid PII SYSEXIT Revenire rapidă de la apelul de sistem PH Instrucțiunea SYSENTER încarcă numărul de la MSR # h în CS, numărul de la MSR # h în EIP, numărul egal cu CS + (selector la următorul descriptor) în SS și numărul de la MSR # în ESP h Această comandă are scopul de a transfera controlul către sistemul de operare - poate fi apelată cu orice CPL, iar codul apelat trebuie să fie în memoria fără segmente cu CPL = De fapt, SYSENTER modifică descriptorii segmentelor utilizate - segmentul de cod va au DPL = , bază , limita de GB va deveni lizibilă și pe de biți, iar segmentul de stivă va primi, de asemenea, baza , limită de GB, DPL = , modul de de biți, acces de citire/scriere și bit de acces a stabilit De asemenea, selectoarele CS și SS primesc RPL = Instrucțiunea SYSEXIT încarcă CS cu MSR # h plus , EIP cu EDX, SS cu MSR # h plus și ESP cu ECX Această comandă este destinată să transfere controlul către modelul de memorie fără segmente cu CPL = și, de asemenea, modifică descriptorii Segmentul de cod primește DPL - , bază , limită de GB, acces de citire, încetează să mai fie slave și devine pe de biți Segmentul de stivă primește, de asemenea, baza , limită de GB, acces de citire/scriere și lățime de biți de de biți Câmpurile RPL din CS și SS sunt setate la Procesoare Intel în modul protejat* Suportul pentru comenzile SYSENTER/SYSEXIT trebuie verificat întotdeauna folosind comanda CPUID (bit ) De asemenea, asigurați-vă că numărul de model al procesorului este de cel puțin trei, deoarece Pentium Pro (Professor Type , Model ) nu are comenzi SYSENTER/SYSEXIT, dar bitul din CPUID este returnat ca SYSENTER execută numai în modul protejat, iar SYSEXIT execută doar cCPL = Intrarea și ieșirea din modul protejat Deci, pentru a comuta în modul protejat, este suficient să setați bitul PE - bit zero în registrul de control CR , iar procesorul va fi imediat în modul protejat Singura cerință suplimentară pe care Intel o face este că în acest moment toate întreruperile, inclusiv NMI-urile, trebuie să fie dezactivate ; pmO asm ; Un program care intră în modul protejat și revine imediat ; Funcționează în modul DOS real și într-o fereastră DOS Windows (interceptările Windows ; excepții care apar atunci când se încearcă trecerea la modul protejat de la V , ; și ne permite să lucrăm, dar numai la nivelul minim de privilegii) ; Compilare: • ; TASM: ; tasm /t pmO asm ; tlink /x /t pmO obj ; MASM: ; ml /c pmO asm ; link pmO obj,,NUL,,, exe bin pmO exe pmO com ; WASM: ; wasm pmO asm ; fișierul wlink pmO obj din DOS COM modelul minuscul cod p Toate exemplele noastre se bazează pe org h ; Acesta este un program COM start: * ; Pregătiți registre de segmente împinge cs pop ds ; DS - segment de date (și cod) al programului nostru apăsați B h pop es ■ ; ES - segment de memorie video Verificați dacă suntem deja în modul protejat mov eax, crO ; Citiți registrul CRO , test al,T ; Verificați bitul PE, jz nr V ; dacă este zero - putem continua, ; în caz contrar, raportați o eroare și ieșiți mov ah, ; Funcția DOS h Intrarea și ieșirea din modul protejat IIIJ J i mov int ret dx,offset v msg; DS'DX - adresă șir h; Ieșire pe afișaj • Sfârșitul programului COM ; (deoarece acesta este modul protejat în care rulează programul nostru DOS, ; atunci trebuie să fie V ) v msg db „Procesor în modul V - nu poate trece la PM$” ; Controlul este transferat aici dacă i ■' mov ax cs mov ds,ax ; Și gata - acum procesorul este în real: mod cu segmente nelimitate Gestionarea întreruperilor și a excepțiilor Până acum, toate programele noastre rulau în modul protejat cu întreruperi complet dezactivate - nu puteau fi controlate de la tastatură, nu puteau funcționa cu discuri și nu făceau nimic altceva decât să citească sau să scrie în anumite zone de memorie Desigur, niciun program nu poate face nimic serios în acest mod - mai devreme sau mai târziu va trebui să ne ocupăm de întreruperi În modul real, adresa operatorului de întrerupere a fost citită de procesor dintr-un tabel situat la adresa în memorie În modul protejat, acest tabel, numit IDT - Interrupt Descriptor Table, poate fi localizat oriunde Este suficient ca adresa și dimensiunea sa să fie încărcate în registrul IDTR Conținutul IDT nu este doar adrese ale handler-urilor, așa cum a fost în modul real, ci descriptori de trei tipuri: poarta de întrerupere, poarta de capcană și poarta de sarcini (formatele de date de descriptor au fost discutate în secțiunea anterioară) Porțile de întrerupere și capcană specifică punctul de intrare al handler-ului, precum și nivelul său de bitness și privilegiu Când controlul este transferat către un handler, procesorul împinge steagurile și o adresă de retur în stivă, la fel ca în modul real, dar după aceea, pentru unele excepții, un cod de eroare suplimentar este împins în stivă, astfel încât nu toți gestionanții pot fi terminat cu un simplu IRETD (sau IRET pentru versiunea pe biți) Singura diferență dintre o poartă de întrerupere și o capcană este că atunci când controlul trece prin poarta de întrerupere, întreruperile suplimentare sunt dezactivate automat până când handlerul execută IRETD Acest mecanism este considerat mecanismul preferat pentru manipulatorii de întreruperi hardware, iar o poartă capcană care nu dezactivează întreruperile în timp ce handlerul se execută este mai bine utilizată pentru a gestiona întreruperile software (care sunt de fapt excepții de tip capcană) În plus, în modul protejat, atunci când este apelat un handler de întrerupere, indicatorul de urmărire TE este resetat În primul rând, să ne uităm la un exemplu de program care gestionează doar o întrerupere hardware de la tastatură utilizând un gateway de întrerupere Pentru a face acest lucru, trebuie să compuneți un IDT, să încărcați adresa sa cu comanda LIDT și să vă amintiți să încărcați ceea ce este conținut în registrul IDTR În modul real, adresa și dimensiunea X , corespunzătoare tabelului vectorial de întrerupere a modului real ; pm asm ; Un program care demonstrează procesarea întreruperilor hardware în modul protejat ; Comută la modul protejat pe de biți și vă permite să introduceți text folosind tastele de la la + Apăsând Backspace șterge caracterul anterior, ; apăsând Esc - ieșiți din program J I Procesoare Intel în modul protejat Compilarea TASM: tasm /t /D TASM pm asm (sau, pentru versiunile x, tasm /t pm asm este suficient) tlink /x / pm obj Compilarea WASM: wasm / pm asm fișierul wlink pm obj formular DOS ; Variații asupra modului în care diferiți asamblatori scriu offset față de de biți ; segment într-o variabilă de biți: ifdef TASM deci equ offset mic; TASM x altfel so equ offset ; WASM endif ; Pentru MASM, se pare, va trebui să adăugați cod suplimentar care convertește ; offset-uri utilizate în IDT r Segmentul RM seg pentru utilizarea publică „CODE” presupune cs:RM seg,ds:PM seg,ss:stack seg start: ; Șterge ecranul movax, int h ; Pregătiți registre de segmente apăsați PM seg pop ds ; Verificați dacă suntem deja în PM: mov edx sgO test al, jz nr V ; Raportați și ieșiți mov dx, deci v msg err exit: muta ah, int h mov ah, Ch int h ; Poate că Windows pretinde că PE = O? nu V : movax, h int Fh test al, al jz no windows Raportați și ieșiți mov dx, deci winjnsg jmp scurt err exit Gestionarea întreruperilor și a excepțiilor ; Deci, suntem cu siguranță în modul real no windows: Calculați bazele pentru toți descriptorii de segment utilizați COG eax, eax mov ax,RM seg shl eax cuvânt mov ptr GDT bitCS+ ,ax ; Baza de bitCS va fi RM seg shr eax, mov byte ptr GDT bitCS+ ,al movax,PM seg shl eax cuvânt mov ptr GDT bitCS+ ,ax ; Baza tuturor celor de biți* va fi cuvânt mov ptr GDT bitSS+ ,ax ; PM seg cuvânt mov ptr GDT bitDS+ ,ax shr eax, mov byte ptr GDT bitCS+ , al mov byte ptr GDT bitSS+ ,al mov byte ptr GDT bitDS+ , al ; Calculați adresa liniară a GDT-ului or eh, eh movax,PM seg shl ex, împinge max adăugați ex, offset GDT mov dword ptr gdtr+ ,eax ; Descărcați GDT Igdt fword ptr gdtr ; Calculați adresa liniară a IDT pop eax B înseamnă că nivelul de privilegiu al lui A este mai mic decât B: □ la încărcarea unui registru DS, ES, FS sau GS, trebuie îndeplinită următoarea condiție: DPL > max(RPL CPL); ' Procesoare Intel în modul protejat □ la încărcarea registrelor SS trebuie îndeplinită următoarea condiție: DPL “ CPL “ RPL; □ în cazul JMP departe, CALL, RET pe segmentul de cod nesubordonat, trebuie îndeplinită următoarea condiție: DPL - CPL (RPL este ignorat); , □ în cazul JMP departe, CALL, RET pe segmentul de cod subordonat, trebuie îndeplinită următoarea condiție: CPL > DPL În acest caz, CPL nu se modifică; □ pentru un CALL departe către gateway-ul de apel trebuie îndeplinite următoarele condiții: CPL DPL al segmentului; □ pentru JMP departe, pe gateway-ul de apel trebuie îndeplinite următoarele condiții: CPL segment DPL dacă este slave, CPL = segment DPL dacă nu este slave Când o procedură este apelată printr-o poartă către un segment de cod nereglementat cu un nivel de privilegii diferit, procesorul efectuează o comutare a stivei Segmentul TSS al sarcinii curente stochează întotdeauna valorile SS:ESP pentru stivele de nivel de privilegiu , I și (dar nu și stiva pentru nivelul de privilegiu , deoarece nu puteți transfera controlul la nivelul decât cu RET/IRET comenzi) La comutarea stivei, parametrii (numărul lor este specificat în descriptorul gateway-ului de apel), steagurile sau codul de eroare (în cazul INT), valorile vechi ale SS:ESP sunt plasate pe noua stivă, înainte de adresa de retur ; pe care comanda RET/IRET îl folosește pentru a comuta înapoi Pentru a reveni dintr-o procedură, RET specifică că RPL-ul selectorului rămas pe stivă este mai mare (mai puțin privilegiat) decât CPL ' Chiar dacă sistemul de operare nu acceptă multitasking, trebuie să emită un segment ȚSS cu SS:ESP valid pentru stive de toate nivelurile -'" dacă va folosi niveluri de privilegii Executarea comenzilor privilegiate Comenzile LGDT, LLDT, LTR, LIDT, MOV CRn, LMSW, CLTS, MOV DRn, INVD, WBINVD, INVLPG, HLT, RDMSR, WRMSR, RDPMC, RDTSC, SYSȘXIT pot fi executate numai dacă CPL este (deși PCE biții și TSD ai segmentului CR permit utilizarea comenzilor RDPMC și RDTSC de la orice nivel) Comenzile LLDT, SLDT, LTR, STR, LSL, LAR, VERR, VERW și ARPL pot fi executate doar în modul protejat - în real și V apare excepția #UD Instrucțiunile CLI și STI sunt executate numai dacă CPL sau » - comutați la dreapta Prioritate medie: | - biți „SAU”; & - pe biți „ȘI”; l - pe biți „SAU exclusiv”; ! - "SAU-NU" pe biți (implicație logică) Cea mai mică prioritate: + - adaos; - scăderea i Asamblator în mediul UNIX Directive de adunare Toate directivele de asamblare UNIX fi întotdeauna încep cu un (punct) Datorită numărului mare de sisteme de operare și de asamblare, au apărut numeroase directive frecvent întâlnite pentru aceștia Să considerăm cele mai utile Directive privind definirea datelor Aceste directive sunt echivalente cu directivele db, dw, dd, df etc utilizate în asamblatoarele DOS/Windows Principala diferență aici este de a da numele unei variabile a cărei valoare este determinată de o astfel de directivă; în asamblatoarele pentru UNIX, este necesar să puneți o etichetă completă care se termină cu două puncte octeți: expresie octet Cuvinte: expresie de cuvânt sau expresie de cuvânt sau expresie scurtă Cuvinte duble: int expresie sau expresie lunga Cuvinte patru (variabile de octeți): expresie quad Variabile de octeți (octa-cuvinte): expresie octa Numere în virgulă mobilă pe de biți: număr flotant sau număr unic Numere în virgulă mobilă pe de biți: numar dublu Numere în virgulă mobilă pe de biți: tfloat număr Șiruri de octeți: ascii șir Șiruri de octeți cu un caracter nul adăugat automat la sfârșit: asciz șir sau string șir Blocuri de date repetate: skip size, value sau space size, value// Umple zonele // de memorie cu dimensiunea specificată // cu octeți // cu valoarea dată Directive de adunare fiii repetare, dimensiune, valoare //Umple o zonă de memorie cu valori // de dimensiunea specificată ( - octeți) // de numărul specificat de ori Mărimea // implicită este , iar valoarea este Variabile neinițializate: Icomm caracter, lungime, aliniere// Rezervă numărul specificat de octeți ȘI pentru un caracter local în secțiunea bss Directive de control al simbolurilor Atribuirea de valori simbolurilor: equ caracter, expresie echiv caracter, expresie set caracter, expresie Gestionarea simbolurilor externe: personaj global sau personaj global simbol extern comm caracter, lungime, aliniere Descrierea simbolurilor de depanare: caracter def endef // Atribuie simbolului valoarea expresiei // La fel ca equ, dar aruncă // un mesaj de eroare dacă simbolul este definit // La fel ca equ, dar puteți // repeta de mai multe ori De obicei, totuși, este mai convenabil // să scrieți pur și simplu „caracter = expresie” // Face simbolul vizibil pentru linker, // și, prin urmare, pentru alte module // programe Și directiva extern este de obicei ignorată -/Și toate caracterele nedefinite sunt considerate // externe // Directiva este echivalentă cu Icomm, dar dacă // un simbol cu acest nume este definit folosind Icomm // într-un alt modul, acesta va fi // folosește simbolul extern // Depanare bloc de descriere simbol Nu vom atinge descrierea simbolurilor de depanare, deoarece formatele acestora variază foarte mult între diferitele sisteme de operare și diferitele formate de fișiere obiect Secțiune Definiție Directive Textul programului este împărțit în secțiuni - cod, date, date neinițializate, simboluri de depanare etc Secțiunile pot fi, de asemenea, împărțite în subsecțiuni, situate imediat una după alta, dar acest lucru este rar folosit subsecţiunea de date Următoarele comenzi vor fi asamblate în secțiunea de date Dacă nu este specificată nicio subsecțiune, datele sunt asamblate în subsecțiunea zero g g g mi Asamblator în mediul UNIX subsecţiunea text Următoarele comenzi vor fi asamblate într-o secțiune de cod nume secțiune, steaguri, @tip sau secțiune „nume”, steaguri Definiția generală a noii secțiuni: □ steaguri (pentru ELF): - w sau #write - scrierea este permisă; - x sau #execinstr - execuția este permisă; - a sau #а ос - este permisă alocarea dinamică a memoriei ( bss); □ tip (pentru ELF): - @progbits - conține date; - @nobits - nu conține date (ocupă doar spațiu) Directive de control al adâncimii de biți codul Următoarele instrucțiuni vor fi asamblate ca instrucțiuni pe biți cod Anulează cod Programe directive de control al pointerului aiidp expresie, expresie, expresie Aliniază indicatorul programului la limita marcată de primul operand A doua expresie specifică ce octeți să completeze secțiunea de ignorare (implicit este zero pentru secțiunile de date și h pentru secțiunile de cod) A treia expresie specifică numărul maxim de octeți pe care această directivă îi poate sări Pe unele sisteme, prima expresie nu este numărul căruia indicatorul ar trebui să devină multiplu, ci numărul de biți din indicator care ar trebui să devină zero când valoare nouă, umplutură Crește indicatorul programului la o nouă valoare în secțiunea curentă Octeții săriți sunt completați cu valorile specificate (zero în mod implicit) Listări de directive de control Interziceți listarea: nolist Permite listarea: listă Sfârșitul paginii: eject Directive de asamblare IIIMIMIHE Dimensiunea paginii ( de rânduri, de coloane în mod implicit): dimensiunea rândurilor, coloanelor Titlul listei: textul titlului Subtitlu: sbttl text Directive de control a asamblarii Includeți textul unui alt fișier în program: include fișier Asamblați blocul dacă condiția este îndeplinită sau dacă un simbol este definit sau nu: if expresie caracter ifdef caracter ifndef sau caracter ifnotdef altfel endif Emite un mesaj de eroare: ou Opriți imediat asamblarea: intrerupere de sarcina Blocuri de repetiție Repetați blocarea programului de numărul specificat de ori: rept numărul de repetări endr Repetați blocul de program pentru toate valorile simbol specificate: caracter igr, adică endr Repetați blocul de program de câte ori există câte octeți în șir, setând caracterul fiecărui octet pe rând: igrs caracter, șir endr În cadrul unui bloc de repetiție, un caracter poate fi referit pornindu-l cu o bară oblică inversă (adică ca un caracter \) De exemplu, un astfel de bloc ip param, , , movl %st( ),%st(\param) endr Asamblator în mediul UNIX ca aceasta irpc param, movl %st( ),%st(\param) endr se asambleaza la: movl %st( ),%st( ) movl %st(O),%st( ) movl %st( ),%st( ) definiții macro Pornire macro: nume macro, argumente Sfârșitul definiției macro: endm Ieșire prematură din definiția macro: exitm În cadrul unei macrocomenzi, un parametru este accesat în același mod ca și blocurile repetate, începând cu o bară oblică inversă Deși directivele standard includ lucruri precum blocuri de repetiție și macro-uri, implementarea lor este destul de simplistă, iar preprocesoare suplimentare sunt adesea folosite în programarea în limbaj de asamblare UNIX Multă vreme s-a obișnuit să se folosească preprocesorul C sau M și mulți asamblatori le pot apela chiar automat, dar proiectul GNU a creat un preprocesor special pentru asamblator - gasp Include diverse extensii ale opțiunilor de asamblare condiționată, construirea buclei, definiții macro, listări, directive de definire a datelor etc Nu ne vom ocupa de implementarea unor astfel de programe complexe care ar putea necesita gasp, nici măcar nu vom folosi jumătate din directivele enumerate, dar despre existența acestui preprocesor trebuie amintit Programare cu libc Toate programele UNIX scrise în C se referă în mod constant la diverse funcții găsite în libc so sau în alte biblioteci standard sau non-standard Programele și procedurile în limbaj de asamblare pot, desigur, să facă același lucru Apelarea unei funcții de bibliotecă se face prin comanda normală caii, iar trecerea parametrilor se face conform convenției C: parametrii sunt împinși pe stivă de la dreapta la stânga și stiva este șters după apelul funcției Singura complicație aici este că unele sisteme, cum ar fi FreeBSD, prefix numele funcției apelate cu un caracter de subliniere, în timp ce altele (Linux și Solaris) nu schimbă numele Dacă numele din sistem sunt modificate, atunci numele procedurilor sunt scrise în asamblator inclusiv main() ar trebui, de asemenea, schimbat în prealabil Programare cu libc Să ne uităm la un exemplu de program care afișează mesajul tradițional Bună lume, cum se face acest lucru !/helloelf s P Un program minimal care tipărește mesajul „Hello world” // Pentru a compila în format ELF U I/ Compilare: // ca -o helloelf o helloelf s // Aspect: string „Bună lume\ ” În cazul FreeBSD, trebuie să faceți doar două modificări - adăugați un caracter de subliniere la începutul numelor puturilor și ale funcțiilor principale și înlocuiți directiva string cu ascii, deoarece versiunea de asamblare distribuită în mod normal cu FreeBSD nu înțelege şir // hellocof s // Program minim care tipărește mesajul „Hello world” // Pentru a compila în varianta FreeBSD a formatului COFF // Compilați pentru „FreeBSD: // ca -o hellocof o hellocof s , Și ld -s -o hellocof bsd /usr/lib/crtO o hellocof o ls | II jnain Assembler în mediul UNIX text globi jnain: mesaje pushl cai „puts popi %ebx ret date mesaj: ascii „Bună lume\ ” Folosind această tehnică, puteți scrie programe exact în același mod ca în G, dar câștigul datorat faptului că asamblarea este permisă pentru a optimiza programul cu câteva procente mai bine decât compilatorul cu C (la optimizare maximă) va fi mic în comparație la portabilitatea pierderii În plus, atunci când scriem orice program semnificativ în întregime în asamblator, ne vom întâlni cu faptul că, ca și în cazul win , va trebui să ne creăm propriile fișiere include cu definiții ale constantelor și structurilor preluate din fișierele include pentru C Și întrucât acești asamblatori nu știu să lucreze cu structurile de date, este necesar să le descriem folosind mijloacele preprocesorului folosit - cpp sau w Cea mai bună utilizare a asamblatorului pentru UNIX (altul decât dezvoltarea efectivă a nucleului de sistem) rămâne totuși cu proceduri nesemnificative care necesită multă putere de calcul - codificare, arhivare, transformări de tip Fourier, care nu sunt foarte complicate și, dacă este necesar, pot fi rescris cu ușurință în asamblator pentru alt procesor sau C Programare fără a utiliza libc Se poate dovedi că programul este forțat să apeleze în mod repetat anumite funcții standard din libc într-o secțiune critică care încetinește execuția întregului program În acest caz, merită remarcat faptul că multe dintre funcțiile libc sunt într-adevăr doar o interfață mai prietenoasă cu C pentru apelurile de sistem furnizate de nucleul sistemului de operare însuși Operațiuni precum I/O, toate operațiunile sistemului de fișiere, operațiunile de proces, operațiunile TCP/IP și așa mai departe, pot fi efectuate prin transferarea controlului direct către nucleul sistemului de operare Pentru a efectua un apel de sistem, trebuie să transmiteți numărul și parametrii acestuia punctului de intrare în nucleu, similar funcției libc syscall( ) Numerele de apel de sistem (situate în /usr/include/sys/syscall h) și modul de acces la punctul de intrare (far caii la : ) sunt standardizate de SysV/ ABL, dar, de exemplu în Linux, o altă este utilizat mecanismul - întrerupere h , prin urmare, se dovedește că accesarea directă a nucleului OS face ca programul să fie legat de un anumit sistem Unele dintre aceste restricții pot fi eliminate" folosind #define corespunzătoare, dar în cazul general, câștigul de viteză Programare fără libc se dovedește a fi o pierdere și mai mare de portabilitate decât utilizarea asamblatorului în UNIX însuși Să vedem cum sunt implementate apelurile de sistem în exemplele prezentate // hellolnx s // Un program care tipărește mesajul „Hello world” pe Linux fără a utiliza lit-c // // Compilați: // ca -o hellolnx o hellolnx s // ld -s -o hellolnx hellolnx o // ■ text globi start start: " // Apelul de sistem # „scriere”, parametrii în Linux sunt plasați de la stânga la dreapta, // în registrele %eax, %ebx, %esx, %edx, %esi, %edi movl$ ,%eax xorl %ebx,%ebx inel %ebx' // %ebx - (identificator stdout) movl $mesaj,%ecx movl $message l,%edx // Transferați controlul către nucleul sistemului - numărul de întrerupere h int $ x // Apel de sistem # „ieșire” (%eax = , %ebx = ) xorl %eax,%eax inel %eax xorl %ebx,%ebx int $ x ■ hli t date mesaj: string „Bună lume\ ” mesaj! = -mesaj Linux este un caz unic în ceea ce privește apelurile de sistem În sistemele UNIX mai tradiționale - FreeBSD și Solaris - apelurile de sistem sunt implementate conform standardului comun SysV/ , iar singura diferență în Programe este că asamblatorul care vine cu FreeBSD nu acceptă unele comenzi și directive // hellobsd s \ , // Un program care imprimă „Hello world” pe FreeBSD fără a utiliza libc // G; în A în C D E FGHIJXLMM p Q în S T u UX z [ \ ] A a b c d e f ff hi J k nnr r h g S t u u X Y z o • -G SSS I r ё i- Іg ■Ц ТІ t „Г іг " l t - I z E - □ t T "t g t □g G me ■і * □g pg © yu a b c A e f g X i j k L m n o p i r s T U zh v y s w e Shch h b TU? L B C A E f G X i j k L m H O P Y R S t U Y V s S E SCH H Orez Codificare KOI -r (RFC ) \ Codarea ISO - este utilizată ca codificare implicită în majoritatea sistemelor de operare comerciale compatibile cu UNIX (vezi Figura ) ABCDEF □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ □ E T> g € s I І J Le n> b k - U c A B c d A F I J k l m n O p R S t u F X c h w sh y y y a b c d A f i j k l m n o p R cu t U V X ts h w y y y y y Nr e b g e S i i І L> Nr h k § U c Orez Codare ISO - Tabelele cu simboluri III Y IG O patru cinci Codificarea cp este utilizată ca codificare principală în aplicațiile grafice pentru Microsoft Windows (vezi Fig ) O A B C D E F 'Tb G • j G Y ■ >> t □ % n> k h c U U J ■'a g II § E © e « „ i o + ti '• g U t e Nr • € » ] SS A B C D E F F I J K L m N O P R S T U V X a b c d A f j j k L m n o p R s t U X ts h sh y y y y y Orez Codificare cf I [IIII Anexa Coduri de caractere ASCII extinse Tabelul Coduri ASCII extinse Cod cheie Cod cheie Cod cheie Cod cheie Cod cheie F Bh Alt-R h Shift-F h Alt-Tab A h Alt-I h F Ch Alt-S Fh Shift-F h Ctrl-Tab h Alt-J h F Dh Alt-T h Alt-O h Alt-Del A h Alt-K h F Eh Alt-U h Alt- h Alt-Sfârșit Fh Alt-L h F Fh Alt-V Fh Alt- h Alt-Home h Ctrl-Dreapta h F h Alt-W lh Alt- h Alt-lns A h Ctrl-End h F h Alt-X Dh Alt- h Alt-PgUp h Ctrl-Home h F h Alt-Y h Alt- h Alt-PgDn A h Ctrl-PgDn h F h Alt-Z Ch Alt- h Alt-Enter Ch Ctrl-PgUp h F h Alt-\ Bh Alt- h Ctrl-F Eh Alt-Up h F h Alt-, h Alt- h Ctrl-F Fh Alt-Jos AOh F h Alt- h Alt- Ah Ctrl-F h Alt-Stânga Bh AH-F h Alt-/ h AltC Bh Ctrl-F h Alt-Dreapta Dh Alt-F h Alt-BS OEh Alt-= Ch Ctrl-F h Alt-K/ A h Alt-F Ah Alt-[ Ah NUL h Ctrl-F h Alt-K* h Alt-F Bh Alt-] Bh Shift-Tab OFh Ctrl-F h Alt-K- Ah Alt-F Ch Alt-; P Ins h Ctrl-F h Alt-K+ Eh Alt-F Dh Alt-' h Del h Ctrl-F h Alt-Kenter A h Att-F Eh Alt-' h SysRq h Ctrl-F h Ctrl-K/ h Alt-F Fh Shift-F h în jos h Ctrl-F h Ctrl-K* h Alt-F h Shift-F h Stânga Bh Ctrl-F Ah Ctrl-K- Eh Alt-FIO h Shift-F h Dreapta Dh Alt-A Eh Ctrl-K+ h Alt-F Bh Shift-F h Sus h Alt-B h Ctrl-K Dh Alt-F Ch Shift-F h End Fh Alt-C Eh Ctrl-K Fh Alt-M h Shift-F h Notă h Alt-D h Ctrl-K h Alt-N h Shift-F Ah PgDn h Alt-E h Ctrl-KO h Alt-O h Shift-F Bh PgUp h Alt-F h Ctrl-K h Alt-P h Shift-F Ch Ctrl-Stânga h Alt-G h Alt-Q h Shift-F O Dh Alt-Esc h Alt-H h Prefixul „K” corespunde tastelor de pe tastatura numerică Tabelele cu simboluri Codurile de scanare de la tastatură Tabelul Codurile de scanare' Cod cheie Cod cheie Cod cheie Cod cheie Esc h Introduceți Ch K* h Ins h ! h Ctrl Dh Alt h Del h @ h A Eh SP h SysRq h # h S Fh Caps Ah Macro h $ h D h F Bh F h % h F h F Ch F h L h G h F Dh PA Ah & h H h F Eh F /LWin Bh * h J h F Fh F /RWin Ch ( OAh K h F h F /Meniu Dh ) OBh L h F h F h Och h F h F h = + ODh i h F h F h BS OEh h F h F h TA OFh Lshift Ah Num h F h O h \l Bh Scroll h F h L lh Z Ch Notă h F h E h X Dh - h F Ah R h c Eh PgUp h F Bh T h V Fh K- Ah EraseEOF Dh Y h până la h Bh Copiere/Redare Fh și h N h K Ch CrSel h I h M h ® Dh Delta h h , h Sfârșit Fh Sens h [{ Ah /? h până la h ] J Bh RShift h PgDn h Prefixul „K” corespunde tastelor de pe tastatura numerică i Anexa Tabelul Coduri de scanare de service Cod Funcție Cod Funcție OOh Buffer de tastatură plin FAh ACK AAh Autotest finalizat FCh Eroare de autotest EOh Prefix pentru tastele gri FDh Eroare de autotest E h Prefix pentru chei fără cod de eliberare FEh RESEND FOh Prefix de eliberare a tastei FFh Eroare de tastatură eeh ecou Anexa Comenzi Intel x Această anexă listează vitezele de execuție a tuturor instrucțiunilor de la procesoarele Intel la Pentium II și codurile de mașină care le corespund Informații generale despre codurile de comandă Format comun de comandă pentru procesor Intel O comandă poate conține până la șase câmpuri: Prefixe - de la zero la patru prefixe de un singur octet Cod - unul sau doi octeți care definesc comanda ModR/M - octet (dacă este necesar) descriind operanzii: biții - : câmp MOD - modul de adresare; biții - : câmpul R/O - fie indică un registru, fie este o continuare a codului de comandă; biții - : câmpul R/M - fie specifică un registru, fie împreună cu MOD - modul de adresare SIB - octet dacă este necesar (extensie ModR/M pentru adresare pe de biți): biții - : S - factor de scalare: biți - : - registru index; biții - : B - registru de bază Offset - , , sau octeți Operand imediat - , , sau octeți - vom folosi /іb și /iw pentru a specifica acești operanzi Valorile câmpului codului de comandă În codurile unor comenzi, vom întâlni biți speciali și grupuri de biți, pe care îi vom nota w, s, d, reg, sreg și cond: □ w = dacă comanda lucrează cu octeți; □ w = dacă comanda funcţionează cu cuvinte sau cuvinte duble; □ s = dacă operandul imediat este complet specificat; □ s = dacă operandul imediat este octetul mic al operandului mai mare și ar trebui tratat ca un număr cu semn; □ d = dacă codul sursă este în câmpul R/O și codul destinație este în R/M; □ d = dacă codul sursă este în câmpul R/M și codul destinație este în câmpul R/O Introducerea dw va însemna că codul de comandă este OOOlOOdw Anexa Câmpul reg specifică registrul de utilizat și are o lungime de biți: OOO - AL/AX/EAX/ST( )/MM /XMM -CL/CX/ECX/ST( )/MM /XMM - DL/DX/EDX/ST( )/MM /XMM -BL/BX/EBX/ST( )/MM /XMM - AH/SP/ESP/ST( )/MM /XMM - CH/BP/EBP/ST( )/MM /XMM - DH/SI/ESI/ST( )/MM /XMM - BH/DI/EDI/ST( )/MM /XMM C r va însemna HOOlreg Câmpul sreg specifică registrul de segment de utilizat: -ES -CS -SS -DS -FS -GS Câmpul cond definește condiția pentru comenzile Jcc, CMOVcc, SETcc, FCMOVcc Semnificațiile sale pentru diferite comenzi: - Despre - NR - C/B/NAE UN - NC/NB/AE - E/Z -NE/NZ -BE/NA -NBE/A -S -NS - R/RE • -NP/PO - L/NGE -NL/GE -LE/NG -LNE/G O intrare ca cc ar însemna OlOOcond Valorile câmpului ModRM Câmpul R/O (biții - ) conține fie trei biți suplimentari ai codului de instrucțiune, fie un cod de operand, care poate fi doar un registru Vom nota al doilea caz prin reg, iar în primul caz vom nota biții folosiți Câmpurile MOD (biții - ) și R/M (biții - ) definesc operandul, care poate fi fie un registru, fie o variabilă în memorie: Comenzi Intel x eu □ MOD = dacă se utilizează adresarea registrului și R/M conține codul registrului reg □ MOD = dacă se utilizează adresarea non-offset ([IN + SI] sau [EDX]) □ MOD = dacă se utilizează adresarea offset pe biți (variabilă[BX+ SI]) □ MOD = dacă se utilizează adresarea offset pe sau de biți Valoarea câmpului R/M este diferită în modurile pe biți și pe de biți R/M în modul biți: - [BX + SI] - [IN + DI] - [BP + SI] - [BP+ DI] - [SI] - [DI] - [ВР] (cu excepția MOD = - în acest caz, după ModR / M există un offset de biți, adică se utilizează adresarea directă) -[VX] R/M în modul pe de biți: - [EAX] - [ESX] - [EDX] - [EVX] - folosit de SIB - [EUR] (cu excepția MOD = - în acest caz, se folosește SIB, după care este localizat offset-ul pe de biți) - [ESI] - [EDI] ■ valorile câmpului SIB Valoarea câmpului S: - nu este folosit - înmulțiți cu - înmulțiți cu - înmulțiți cu Valorile câmpurilor I și B: (I - registru folosit ca index, adică înmulțit cu S; B - registru de bază, care nu se înmulțește) - EAX - ECX -EDX -EVX II Anexa - pentru I - fără indice j' pentru B - ESP - pentru I - EUR pentru B - EUR numai dacă MOD = sau ; dacă MOD = - fără bază -ESI -EDI Câmpurile ModR/M și SIB vor fi scrise ca /r dacă câmpul R/O conține un cod de registru sau / - / dacă câmpul R/O conține încă trei biți de cod de comandă În alte cazuri, câmpurile ModR/M și SIB lipsesc doar pentru instrucțiunile fără operanzi, deci nu vor fi indicați suplimentar Informații generale despre viteze de execuție Vitezele de execuție a instrucțiunilor pentru procesoarele - P sunt date în cicluri de clock (atunci când se spune că viteza de clock a procesorului este de MHz, asta înseamnă că trec de milioane de cicluri pe secundă) Pentru procesoarele P (Pentium, Pentium MMX), pe lângă Speed, este indicat dacă comanda poate fi executată simultan cu altele și, dacă da, în ce jon-way (vezi secțiunea ): ѵ ; □ UV - se poate executa simultan, în orice conductă; □ PU - poate fi executat simultan, în U-pipeline; DIN □ РV - poate fi executat simultan, într-o conductă în V; □ FX - poate fi executat simultan cu comanda FXCH; □ NP - nu poate fi executat simultan (pentru MMX - nu poate fi executat simultan cu o comandă de același tip, care este indicată după litera p) Pentru procesoarele P (Pentium Pro, Pentium II) este indicat numărul de microoperații pentru care este decodificată comanda Comenzile cu o structură complexă sunt marcate cu litera C (vezi secțiunea ) În toate cazurile, sunt date vitezele minime posibile - dacă magistrala de date nu este blocată, operanzii sunt aliniați pe granițele de cuvinte duble, operanzii sunt în memoria cache de date, instrucțiunile de la adresa pentru a sări sunt localizate în memoria cache de coduri, salturile sunt ghicite corect de procesor, memoria cache nu este umplută în momentul executării instrucțiunii, paginile sunt în TLB (altfel ar trebui adăugate - de cicluri pentru P ), nu există excepții în momentul executării comenzii, nu apare AGI etc Operanzii sunt notați după cum urmează; □ im - operand imediat; □ І , ііб, І - operand direct de dimensiunea specificată; □ ac - EAX, AX, AL; □ d - orice registru de uz general; g - AN, AL, VN, BL, DH, DL, CH, CL; yg- J g - AX, BX, CX, DX, BP, SP, SI, DI; ȘI • Comenzi Intel x G p eu □ r - EAX, EBX, ECX, EDX, EUR, ESP, ESI, EDI; □ sr - registru segment; □ m - operand în memorie; □ mm - registru MMX; □ xmm - registru SSE; □ sO - registrul ST(O); □ si - registrul ST(i) Pentru instrucțiunile de săritură condiționată, sunt date două viteze de execuție - dacă săritura a avut loc și dacă nu Alte abrevieri folosite: □ RM - mod protejat; □ RM - mod real; □ Mod VM - V ; □ TS - comutare sarcini; □ CG - gateway de apel; □ TG - poarta sarcinilor; □ /ip - creşterea privilegiilor; □ /out - reduce privilegiile; □ /NT - comutați la o sarcină imbricată Timp de schimbare a sarcinii: □ Pentium: TS - când este comutat la TSS pe și biți și când este comutat la TSS V ; □ : TS - la trecerea la TSS pe și biți și la trecerea la TSS V ; □ : TS - - la trecerea la TSS pe și biți și - la trecerea la V ; □ : TS = Prefixe Toate prefixele sunt executate într-un ciclu și au o dimensiune de octet: OFOh: BLOCARE F h: REPNE/REPNZ F h: REP/REPE/REPZ Eh:CS: h: SS: Eh:DS: h: ES: h:FS: h: GS: h:OS h: AS j IG II SALUT Anexa Instrucțiuni pentru procesoare Intel - Pentium III Tabelul Comenzi Cod de comandă P P AAA NP m AAD І D ib NP m AAM І D ib NP m AAS F NP m ADC ac,im w im PU m ADC g,im sw/ im PU ' m ADC m,im sw/ im +ea PU m ADC r,r dw /r PU m ADC m,r dw /r '+ea PU m ADC r,m dw /r +ea PU m Adăugați ac,im w im UV m ĂDD r,im sw/ im UV m ADD m,im sw/ im +ea UV m ADD r,r OOdw /r UV m ADD m,r OOdw /r +ea UV m ADD r,m OOdw /r +ea UV m ȘI ac,im w im UV m AND r,im sw/ im UV Im AND m,im sw / im +ea UV m ȘI r,r dw /r UV m AND m,r dw /r +ea UV m AND r,m dw /r +ea UV m ARPL r,r /r NP C ARPL m,r /r NP C LEGAT r,m /r NP C BSFr ,r OF VS /r + n NP m BSF r ,r OF BC /r + n NP m BSF r ,m OF VS /r +Zp NP m BSF r ,m OF VS /r +Zp NP m BSR r ,r OF BD /r +Zp NP m BSR r ,r OF BD /r + n NP m BSR r ,m OF BD /r + n NP m BSR r ,m OF BD /r + n NP m BSWAP r OF C r NP m W r,r OF AZ /r NP m BT m,r OF AZ /r NP c W r,i OF BA / ib NP m BT m,i OF BA / ib NP m VTS g, g OF BB /r NP m BTC t,g OF BB /r NP s VTS g,і OF BA / ib NP m Comenzi Intel x Tabelul Comenzi (continuare) Cod de comandă P P ВТС t,і OF BA / ib • NP m BTR g,g OF B /r NP m BTR t,g OF B /r NP s BTR G,І OF BA / ib NP m BTR m,i OF BA / ib NP m BTS r, r OF AB /r NP m BTS m,r OF AB /r NP c BTS r,i OF BA / ib NP m BTS m,i OF BA / ib NP m Apelați lângă im E im PV m Apelați lângă r FF / NP C CALL lângă m FF/ +ea NP C Apelați departe im (RM) A im NP c Apelați departe im (PM) A im NP c CALL im (CG) A im NP c CALL im (CG/in) A im NP c Apelați im (TS) A im TS +TS +TS NP c Apel, im (TG) A im TS +TS +TS NP c CALL far m (RM) FF / +ea NP c CALL far m (PM) FF / NP c APEL m CG FF/ NP c APEL m CG/in FF/ NP c APEL m TS FF/ TS +TS +TS NP c APEL m TG FF/ TS +TS +TS NP c CBW NP m CDQ NP m CLC F NP m CLD FC NP m CLI FA NP C CLTS DE NP C CMC F NP m CMOVcc r,r OF cc /r m CMOVcc r,m OF cc /g m CMP ac,im Cw im UV m CMP r,im sw/ im UV m CMP m,im sw/ im +ea UV m ] Li Anexa Tabelul Comenzi (continuare) Cod de comandă P P CMP g,g dw /r UV m CMP m,r dw /r +ea UV m CMP r,m dw /r +ea UV m CMPSB A NP C REP* CMPSB F / A + n + n + n + n + n + n NP C CMPSW/ CMPSD A NP c REP* CMPSW/D F / A + n + n + n + n + n + n NP c CMPXCHG r,r OF Bow /r NP c CMPXCHG m,r OF Bow /r , NP >C CMPXCHG B m OF C / NP c CPUID FA NP C CWD NP m CWDE Ș, ■ NP m DAA NP , m DAS F , NP im DEC r / r • UV Ipi DEC r FEw/ UV Trn DEC m FEw/ +ea UV m DIV r F w / NP m DIV r F w / NP m D(V r F w / NP m DIV m F w / +ea , NP m DIV m F w / O +ea NP m DIV m F w / • NP m EMMS DE NP C ENTER I C iw ib NP c ENTER C iw ib ; NP C ENTER І І C iw ib (m=n- ) + m + m + m + n + nNP C F XM D FO - , NP c FABS D E FX im FADD m D / +ea , / FX m FADD m DC/ +buc / FX m FADD sO,și D COR / FX Im FADD si,sO DC COr / FX im FADDP si,sO DE COr l: / FX m FBLD m DF/ +ea - NP iC, Comenzi Intel x P III Tabelul Comenzi (continuare) Cod de comandă P P FBSTP gp DF/ +ea NP C FCHS D E FX m FCLEX B DB E NP m FCMOVB sO Si DACOr m FCMOVE sO Si DA C r m FCMOVB E soO si DADOR m FVMOVU sO si DA D r m FCMOVNB sO si DB COR m FCMOVNE sO si DB C r m FCMOVNBE sO si D DOr m FCMOVNU sO si DB D r ■ m FCOM m D / +ea • / FX m FCOM m DC/ +ea / FX m FCOM si D DOr / FX m FCOMP m D / +ea / FX m FCOMP m DC/ +ea / FX m FCOMP si D D r / FX rn FCOMPP DE D / FX m FCOMI sO si DB PENTRU m FCOMIP so si DFFor m FCOS D FF NP c FDECSTP D F NP m FDIV m D / +ea FX m FDIV m DC/ +ea FX m FDIV sO si D PENTRU FX m FDIV si sO DC F r FX m FDIVP si sO DE F r FX m FDIVR m D / +ea FX m FDIVR m DC/ +ea FX m FDIVR so si D F r FX m FDIVR si sO DC PENTRU FX m FDIVRP si sO DE FOR FX m FDISI B DB E FNOP FENI B DB EO FNOP FFREE si DDCor NP m I ILI Anexa Tabelul Comenzi (continuare) Cod de comandă P P FIADD t DA/ O +ea / NP C FIADD t DE/ O +ea / NP c FICOM t DE/ +ea / NP c FICOM t DA/ +ea / NP c FICOMPm DE/ +ea / NP c FICOMP t DA/ O +ea / NP c FIDIVm DA / +ea NP c FIDIV t DE/ O +ea NP c FIDVR t DA/ +ea NP c FIDVR t DE/ +ea NP c FILD t DF/ +ea / NP m FILD t DB/ +ea / NP m FILD t DF/ +ea / NP m FIMULm DA/ +ea / NP C FIMUL t DE / O +ea / NP C FINCSTP D F NP m FINIT B DB E NP c FIST t DF/ +ea NP m FIST t DB / +ea NP m FISTP t DF/ +ea NP m FISTP t DB/ +ea NP m FISTP t DF / +ea NP m FISUB t DE/ O +ea / NP C FISUB t DA/ O +ea / NP C FISUBR t DE / +ea / NP C FISUBR t DA / O +ea / NP C FLD t D / +ѳа FX m FLD t DD / +ea FX m FLD t DB / +ea NP m FLD si D COR FX m FLD D E NP m FLDL T D E / NP m FLDL E D EA / NP m FLDPI D EB / NP m FLDLG D EU / NP m FLDLN D ED / NP m FLDZ D EE NP m FLDCWm D / +ea NP m FLDENV m D / +ea NP C FLDENVT (PM) D / NP C Comenzi Intel x eu! eu Tabelul Comenzi (continuare) Cod de comandă P P FMUL m D / O +ea / FX m FMUL t DC/ +ea / FX m FMUL soO si D C r O +ea / FX m FMUL si,sO DC C r O +ea / FX m FMULP si,sO DE C r +ea / FX m FNCLEX DB E - / NP m FNDISI DB E i FNOP FNENI DB EO FNOP FNINIT DB E NP C i FNOP D DO NP FNSAVE m (RM) DD / +ea NP c FNSAVE m (PM) DD / NP c FNSETPM DB E FNOP FNSTCW m D / NP m FNSTENV m D / +ea NP C FSTSWm B DD / NP m FSTSW AX B DF EO NP m FPATAN D F NP C FPREM D F NP C FPREM D F NP C FPTAN D F NP C FRNDINT D FC NP C FRSTOR m (RM) DD/ +ea NP C FRSTOR m (PM) DD / NP C FSAVE m (RM) B DD / +ea NP C FSAVE m (PM) B DD / NP C FSCALE D FD NP C FSETPM B DB E FNOP FSIN D FE NP C FSINCOS D FB NP C FSQRT D FA NP m FST m D / +ea NP m FST m DD / +ea NP m FST si DD DOr NP m Legea jj | | ȚI Anexa Tabelul Comenzi (continuare) Cod de comandă P P FSTP m D / +ea NP m FSTP t DD/ +ea NP m FSTP t DB / +ea NP C FSTP si DD D r NP m FSTCWm B D / NP m FSTENV m B D / +ea NP C FSTSWm B DD / NP m FSTSW AX B DF EO NP m FSUB m D / +ea / FX m FSUB m DC/ +ea / FX m FSUB sO,si D EOr / FX m FSUB si,sO DC E r / FX m FSUBP si sO DE E r / FX m FSUBR m D / +ea / FX m FSUBR m DC / +ea / FX m FSUBR sO,si D E r / FX m FSUBR si,sO DC EOr / FX ni FSURP si sO DE EOr / FX t FTST D E / FX m FUCOM Si DD EOr / FX t FUCOMP si DD E r / FX m FUCOMPP DA E / FX m FUCOMI sO si DB E r m FUCOM IP sO,si DF E r m FWAIT B NP m FXAM D E NP m FXCH si D C r PV m FXRSTOR m DE AE / s FXTRACT D F NP c FXSAVEm OF AE / s FYL X D F NP c FYL XP D F NP c HLT F NP c IDIV r F w / NP m IDIV r F w / NP m IDIV r F w/ NP m IDIV m F w / O +ea NP m IDIV m F w / , +ea NP m III II Comenzi Intel x Tabelul Comenzi (continuare) Cod de comandă P P IDIV t F w/ NP m IMUL g F w / NP m IMUL- F w/ NP m IMUL r F w/ NP m IMUL t F w/ +ea NP m IMULm F w/ +ea NP m IMUL r F w/ NP m IMUL g,g,І B /r ib NP m IMUL r ,r ,i /r im NP m IMUL r ,r ,i /r im NP m IMUL g,g,I B /r ib NP m IMUL r ,m ,i /r im NP m IMUL g ,t ,і /r im NP m IMULH , r OF AF /r NP m IMUL r ,r OF AF /r NP m IMUL- , m OF AF /r NP m IMUL r , m OF AF /r NP m IN ac,i (RM) E w ib NP C IN ac,i (CPLCIOPL) E w ib NP C IN ac,i (CPLXOPL) E w ib NP C IN ac,i (V ) E w ib NP C IN ac,DX (RM) ECw NP C IN ac,DX (CPLCIOPL) ECw NP C IN ac,DX (CPL>|OPL) ECw NP C IN ac,DX (V ) ECw NP C INC r FEw/ UV m INC P / r UV m INC m FEw/ +ea ' UV m INS* (RM) Cw NP C Anexa Tabelul Comenzi (continuare) Cod de comandă P P INS* (CPKIOPL) Cw NP c INS* (CPL>IOF'L) Cw NP c INS* (V ) Cw NP c INT І (RM) CD ib NP m INT І (RM) CD ib NP C INT I (PM/in) CD ib NP C INT (V ) CD ib NP C INT І (TG) CD ib TS +TS +TS NP C INT (RM) CC NP C INT (PM) CC NP C INT (PM/in) CC NP C INT (V ) CC NP C INT (TG) CC TS +TS +TS NP C ÎN (OF= ) CE NP C INTO (RM) CE NP C ÎN (PM) CE NP C INTO (PM/in) CE NP C INTO (V ) CE NP c INTO (TG) CE TS +TS +TS NP c INVD DIN NP C INVLPG m OF / NP c IRET/IRETD (RM) CF NP c IRET/IRETD (PM) CF NP c IRET/IRETD (PM/out) CF NP c RET/IRETD (PM/NT) CF TS +TS +TS c Jcc І (neeliberat) c ib ' PV m Jcc І (număr) c ib PV m Jcc im (neeliberat) OF c ib PV m Jcc im (emisiune) OF c ib PV m JCXZ І (neeliberat) E ib NP m JCXZ i (problemă) E ib NP m JMP lângă I EB ib PV m JMP lângă i / E ib PV m Comenzi Intel x Tabelul Comenzi (continuare) Cod de comandă P P JMP peag g FF / NP m JMP lângă m FF / +ea NP m JMP departe im (RM) EA im NP C JMP departe im (PM) EA im NP C JMP departe im (CG) EA im NP C JMP departe im (TS) EA im TS +TS , +TS NP C JMP departe im (TG) EA im TS +TS +TS NP C JMP far m (RM) FF/ +ea NP C JMP far m (PM) FF/ NP C JMP far m (CG) FF / NP C JMP far m (TS) FF / +TS +TS +TS NP C JMP far m (TG) FF/ +TS +TS +TS NP C LARF F NP m LAR r,r OF /r NP c l-AR r,m OF /r NP c LDS r,m (RM) C /r +ea NP c LDS r,m (PM) C /r NP c LEA r,m D/r +ea UV m LASĂ C NP m LES r,m (RM) C /r +ea NP C LES r,m (PM) C /r NP C LFS r,m (RM) OF B /r NP c LFS r,m (PM) OF B /r NP c LGDTm OF / NP c LGS r,m (RM) OF B /r NP c LGS r,m (PM) OF B /r NP c LIDTm OF / NP c LLDTr OF / NP c LLDTm OF / NP c LMSWr OF / NP c LMSWm OF / NP c BLOCARE FO NP c I și WII Anexa Tabelul Comenzi (continuare) Cod de comandă P P LODSB/ LODSW/ LODSD ACw NP m LOOP І (neeliberat) E ib NP m LOOP І (problemă) E ib NP m LOOPE i (nu este lansat) E ib NP m BUCLA І (problemă) E ib NP m LOOPNE i (epuizat) EO ib NP m LOOPNE I (număr) EO ib NP m LSL g,g OF /r sau NP C LSL r,m OF /r sau NP C LSS r,m (RM) din B /r NP C LSS r,m (PM) OF B /r NP C LTR r OF / NP C LTR m OF / NP C MOV r,r dw /r UV m MOV m,ac AOdw im UV m MOV m,r dw /r +ea UV m MOV ac,m AOdw im UV m MOV r,m dw /r +ea UV m MOV m,im C w / im +ea UV m MOV r ,i BOr ib UV m MOV r / , / B r ib UV m MOV sr,r E/r NP m MOV sr,r (PM) E/r NP m MOV sr,m E/r +ea NP m MOV sr,m (PM) E /r NP m MOV r,sr C /r NP m MOV m,sr C/r +ea NP m MOV CRO t OF /g NP s MOV CR ,r OF /r NP c Comenzi Intel x III II II Tabelul Comenzi (continuare) Cod de comandă P P MOV CR ,r OF /r NP C MOV CR ,r OF /r NP c MOV r,CRx OF /r NP c MOV DR - ,r OF /r NP c MOV DR - ,r OF /r NP c MOV DR - ,r OF /r NP c MOV r,DR - OF /r NP c MOV r,DR - OF /r NP c MOV r,DR - OF /r NP c MOV TR ,r OF / MOV TR ,r OF / MOV TR ,r OF / MOV r,TR DIN / MOV r,TR DIN / MOV r,TR DIN / MOVD mm,rm OF E /r PU m MOVD r ,mm OF E /r PU m MOVD m ,mm OF E /r PU m MOVQ m ,mm OF F /r PU m MOVQ mm,m OF F /r PU m MOVQ mm mm OF *F /r PU m MOVS* A w NP c MOVSX r,r OF BEw /r NP m MOVSX r,m OF BEw /r NP m MOVZX rr OF B w /r NP m MOVZX r,m OF B w /r NP m MUL r F w/ NP m MUL r F w/ NP m MUL r F w / NP m MUL m F w/ +ea NP m MULm F w/ +ea NP m MUL m F w / NP m NEG r F w/ NP m NEG m F w/ +ea NP m NOP UV m II Gі li Anexa Tabelul Comenzi (continuare) Cod de comandă P P NU g F w / NP m NU m F w / +ea NP m SAU ac,im OCw im UV m SAU g,im sw/ im UV m SAU m,im sw / im +ea UV m SAU r,r dw /r UV m SAU m,r dw /r +ea UV m SAU r,m dw /r +ea UV m OUT i ,ac (RM) E w ib NP C OUT i ,ac (CPL IOPL) E w ib NP C OUT i ,ac (CPL IOPL) EEw NP C OUT DX,ac (V ) EEw NP C OUTS* (RM) Ew NP C OUTS* (CPKIOPL) Ew NP C OUTS* (CPL>IOPL) Ew NP C IEȘIRI* (V ) Ew NP C PACKSSWB mm,mm OF /r UVNP m PACKSSWB mm,m OF /r PU NP m PACKSSDW mm,mm OF B /r UV NP im PACKSSDW mm,m DE B /r PU NP m PACKUSWB mm,mm OF /r UV NP m PACKUSWB mm,m OF /r PU NP m Comenzi Intel x Tabelul Comenzi (continuare) Cod de comandă P P PADDB mm,mm OF FC /r UV m PADDB mm,m OF FC /r PU m PADDW mm,mm OF FD /r UV m PADDW mm,m OF FD /r PU m PADDD mm,mm OF FE /r UV m PADDD mm,m OF FE /r PU m PADDSB mm mm DE EU /g UV m PADDSB mm,m DE EU /g PU m PADDSW mm,mm DE ED /g UV m PADDSW mm,m OF ED /r PU m PADDUSB mm,mm OF DC /r UV m PADDUSB mm,m OF DC /r PU m PADDUSW mm,mm OF DD /r UV m PADDUSW mm,m OF DD /r PU m PAND mm mm OF DB /r UV m PAND mm,m OF DB /r PU m PANDN mm mm OF DF /r UV m PANDN mm,m OF DF /r PU m PCMPEQB mm,mm OF /r UV m PCMPEQB mm,m OF /r PU m PCM PEQW mm,mm OF /r UV m III ■III Anexa Tabelul Comenzi (continuare) Cod de comandă P P PCMPEQW mm,m OF /g PU m PCMPEQD mm mrn OF /r UV m PCMPEQD mm,m OF /r PU m PCMPGTB mm,mm OF /r UV m PCMPGTB mm,m OF /r PU m PCMPGTW mm,mm OF /r UV m PCMPGTW mm,m OF /r PU m PCMPGTD mm,mm OF /r UV m PCMPGTD mm,m OF /r PU m PMADDDWD mm,mm OF F /r UVNP m PMADDDWD mm,m OF F /r PU NP m PMULHW mm,mm OF E /g UVNP m PMULHW mm,m OFE /r • PU NP m PMULLW mm,mm OF D /r UVNP m PMULLW mm,m FD /r PU NP m POP r r UV m POP m F/ +ea NP ,C POP DS (RM) F NP C POP DS (PM) F NP C POP ES (RM) NP C POP ES (PM) NP C POP SS (RM) NP C POP SS (PM) NP C POP FS (RM) FA NP C POP FS (PM) FA NP C POP GS (RM) OF A NP C POP GS (PM) FA NP C Comenzi Intel x GBP Tabelul Comenzi (continuare) Cod de comandă P P POPA/POPAD NP C POPF/POPFD (RM) D NP c POPF/POPFD (PM) D NP c POR mm,mm OF U /r UV m POR mm,m OF U /r PU m PSLLW mm,mm OF F /r UV NP m PSLLW mm,m OF F /r PU NP m PSLLW mm,i OF / ib UV NP m PSLLD mm,mm OF F /r UV NP m PSLLD mm,m OF F /r PU NP m PSLLD mm,i OF / ib UV NP m PSLLQ 'mm,mm OF F /r UV NP m PSLLQ mm,m OF F /r PU NP m PSLLQ mm,i OF / ib UV NP m PSRAW mm,mm OF E /r UV NP m PSRAW mm,m OF E /r PU NP m PSRAW mm,i OF / ib UV NP m PSRAD mm,mm OF E /r UV NP m PSRAD mm,m OF E /r PU NP m PSRAD mm,i OF / ib UVNP m PSRLW mm,mm OF D /r UV NP m PSRLW mm,m OF D /r PUNP m PSRLW mm,i OF / ib UV NP m PSRLD mm,mm OF D /r UV NP m PSRLD mm,m OF D /r PU NP m I Anexa Tabelul Comenzi (continuare) Cod de comandă P P PSRLD mm,i OF / ib UV NP m PSRLQ mm,mm OF D /r UV NP m PSRLQ mm,m OF D /r PU NP m PSRLQ mm,i OF / ib UV NP m PSUBB mm,mm OF F /r UV m PSUBB mm,m OF F /r PU m PSUBW mm,mm OF F /r UV m PSUBW mm,m OF F /r PU m PSBD mm,mm OF FA/r UV m PSUBD mm,m OF FA /r PU m' PSUBSB mm mm OF E /r UV m PSUBSB mm,m OF E /r PU m PSUBSW mm,mm FE /r UV m PSUBSW mm,m OF E /r PU m PSUBUSB mm,mm OF D /r UV m PSUBUSB mm,m OF D /r PU m PSUBUSW mm,mm OF D /g UV im PSUBUSW mm,m OF D /r PU m PUNPCKHBW mm,mm OF /r UVNP m PUNPCKHBW mm,m OF /r PU NP m PUNPCKHWD mm,mm OF /r UVNP m Comenzi Intel x Tabelul Comenzi (continuare) Cod de comandă P P PUNPCKHWD mm,m OF /r PU NP m PUNPCKHDQ mm mrn OF A /r UV NP m PUNPCKHDQ mm,m OF A /r PU NP m PUNPCKLBW mm mm OF /r UV NP m PUNPCKLBW mm,m OF /r PU NP m PUNPCKLWD mm,mm OF /r UV NP m PUNPCKLWD mm,m OF /r PU NP m PUNPCKLDQ mm,mm OF /r UV NP m PUNPCKLDQ mm,m OF /r •i PU NP m PUSH r r UV m PUSH m FF / +ea NP m IMPINGERE I A ib NP m PUSH i / im NP m IMPINGERE CS OE NP m IMPINGERE DS E NP m IMPINGERE ES NP m IMPINGERE SS NP m IMPINGERE FS DE AO NP m IMPINGERE GS A NP m PUSHA/ PUSHAD NP C PUSHF/ PUHFD(RM) C NP C PUSHF/ PUHFD(PM) C NP C PXOR mm,mm OF EF /r UV m PXOR mm,m OF EF /r PU m RCL r, D w/ PU m RCLm, DOw / +ea PU m RCL r,CL D w/ + n +n +n NP C IIIIIII eu Anexa Tabelul Comenzi (continuare) Cod de comandă P P RCLm CL D w/ +ea+ n +n +n NP C RCL g,і COw / ib +n +n NP c RCL m,i COw / ib +n +n NP c RCR r, D w/ PU m RCRm, D w/ +ea PU m RCR r,CL D w/ + n +n +n NP c RCR m,CL D w/ +ea+ n +n +n NP C RCR r,i COw / ib +n +n NP C RCR m,i COw / ib +n +n NP C RDMSR DE ' NP C RDPMC din C RDTSC DIN NP C RETN C NP m RETN І C iw / NP C RETF (RM) CB NP m RETF (PM) CB ? NP C RETF (PM/out) CB • NP C RETFi (RM) CA iw NP C RETFi (PM) CA iw NP C RETFi (PM/out) CA iw NP C R Lr, D w/ PU m R Lm, DOw / +ea PU m ROL r,CL D w/ + n +n +n NP m ROL m,CL D w/ +ea+ n +n +n NP m ROL r,i COw / ib +n +n PU m ROL m,i COw / ib +n +n PU m ROR r, D w/ PU m ROR m, DOw/ +ea PU m ROR r,CL D w/ + n +n +n NP tm ROR m,CL D w/ +ea+ n +n +n NP m ROR r,i C w/ ib +n +n PU m ROR m,i C w/ ib +n +n PU m RSM OFAA NP C SAHF E NP m SALr, DOw / PU m SALm, DOw / +ea PU m SAL r,CL D w/ + n +n +n NP Im SAL m,CL D w/ +ea+ n +n +n NP m SAL r,i COw / ib +n +n PU m Comenzi Intel x I Tabelul Comenzi (continuare) Cod de comandă P P SAL m,i COw / ib +n +n PU m SALC D ? ? NP? SAR r, D w/ PU m SAR m, D w/ +ea * PU , m SAR r,CL D w / + n +n +n * NP m SAR m,CL D w/ +ea+ n +n +n NP m SAR r,i COw / ib +n +n PU m SAR m,i COw / ib +n +n PU m SBB ac,im Cw im PU m SBB r,im sw/ im PU m SBB m,im sw/ im +ea PU m SBB r,r dw/r PU m SBB m,r dw /r +ea PU m SBB r,m dw /r +ea PU m SCASB/ SCASW/ SCASD AEw NP m REP* SCAS* F / AEw + n + n + n + n + n + P NP C SETcc r (emisiune) OF cc NP m SETcc r (NU) DE cc NP m SETcc m (rel ) OF cc NP m SETcc m (NU) DE cc NP m SGDT m OF / NP m SHLr, D w/ PU m SHLm, D w/ +ea PU m SHL r,CL D w / + n +n +n - NP m SHLm CL D w/ +ea+ n +n +n NP m SHLr,i COw / ib +n +n PU m SHL m,i COw / ib +n +n PU m SHR r, D w/ PU Im SHR m, D w/ +ea PU m SHR r,CL D w/ + n +n +n NP m SHR m,CL D w/ +ea+ n +n +n NP m SHR r,i COw / ib +n +n PU m SHR m,i COw / ib +n +n PU m SHLD r,r,im FA NP m SHLD rrCL OF A NP m ■MII Anexa Tabelul Comenzi (continuare) Cod de comandă P P SHLD m,r,im OF A NP m SHLD m,r,CL FA NP m SHRD r,r,im OF AC NP m SHRD r,r,CL OF AD NP m SHRD m,r,im OF AC NP m SHRD m,r,CL OF AD NP m SIDTm OF / NP C SLDT r OF / NP m SLDT m OF / NP C SMSW r OF / NP m SMSWm DIN / NP C STC F NP m STD FD NP m STI FB NP C STOSB/ STOSW/ STOSD AAw NP m REP STOS* F AAw + n + n + n + n + n +n NP C STR r OF / NP m STR m OF / NP C SUB ac,im Cw im UV m SUB r,im sw / im UV m SUB m,im sw/ im +ea UV m SUB r,r dw /r ' UV m SUB m,r dw /r +ea UV m SUB r,m dw /r +ea UV m SISTEMA DE C SYSEXIT DE C TEST ac,im A w im UV m TEST r,im F w / im UV m TEST m,im F w / im +ea UV m TEST r,r w /r UV m TEST m,r w /r +ea UV m TEST r,m w/r +ea UV m UD OFOB C VERR r OF / NP C VERR m OF / NP C VERW r OF / NP C VERWm DIN / NP C Așteptați B NP m Comenzi Intel x IIIII Tabelul Comenzi (sfârșit) Cod de comandă P P WBINVD DIN + NP c WRMSR DE NP c XADD g,g OF COw /r NP m XADD m,r OF COw /r NP C XCHG ac,g/g,ac r NP m XCHG g,g w /r NP m XCHG g,t/t,g w /r +ea NP C XLAT D NP m XOR ac,im w im UV m XOR g,im sw/ im UV m XOR m,im sw/ im +ea UV m XOR r,r dw /r UV m XOR m,r dw /r +ea UV m XOR r,m dw /r +ea UV m Tabelul Codurile de comandă de extensie SSE Cod de comandă MOVAPS xmm,xmm/m DIN /r MOVAPS xmm/m ,xmm OF /r MOVUPS xmm,xmm/m DIN /r MOVUPS xmm,xmm/m OF /r MOVLPS xmm,m OF /r MOVLPS m xmm F /r MOVHLPS xmm xmm OF /r MOVHPS xmm,m OF /r MOVLHPS xmm xmm F /r MOVHPS m ,xmm ' OF /r MOVSS xmm,xmm/m F F /r MOVSS xmm/m ,xmm F OF /r ADDPS xmm,xmm/m OF /r ADDSS xmm,xmm/m F OF /r SUBPS xmm,xmm/m OF C /r SUBSS xmm,xmm/m F OF C /r MULPS xmm,xmm/m DIN /r MULSS xmm,xmm/m F OF /r DIVPS xmm,xmm/m OF E /r DIVSS xmm,xmm/m F OF E /r I D III li Anexa Tabelul Codurile de comandă de extensie SSE (sfârșit) Cod de comandă RSQRTPS xmm,xmm/m OF /r RSQRTSS xmm,xmm/m F OF /r SQRTPS xmm,xmm/m OF /r SQRTSS xmm,xmm/m F OF /r RCPPS xmm,xmm/m OF /r RCPSS xmm,xmm/m F OF /r MAXPS xmm,xmm/m OF F/r MAXSS xmm,xmm/m F F F /r MAXPS xmm,xmm/m OF F /r MAXSS xmm,xmm/m F OF F /r MINPS xmm,xmm/m OF D /r MINSS xmm,xmm/m F OF D /g CMPPS xmm,xmm/m ,i OF C /r i CMPSS xmm,xmm/m ,i F OF C /ri COMISS xmm,xmm/m OF F/r UCOMISS xmm,xmm/m OF E /r CVTPI PS xmm,mm/m F A/r CVTSI SS xmm,r/m F OF A /r CVTPS PI mm,xmm/m OF D /r CVTSS SI r ,xmm/m F OF D /r CVTTPS PI mm,xmm/m OF C /r CVTTSS SI r ,xmm/m F OF C /r ANDPS xmm,xmm/m DIN /r ANDNPS xmm,xmm/m OF /r ORPS xmm,xmm/m OF /r Abrevieri folosite Interfață binară a aplicației ABI AGI Address Generation Interlock AMIS Alternative Multiplex Interrupt Specification Interfața programului de aplicație API Codul standard american ASCII pentru schimbul de informații AT&T American Telephone and Telegraph Decimală codificată binar BCD Sistem de intrare/ieșire BIOS de bază BIT Cifră binară BPB BIOS Parameter Block BRM Big Real Mode BSD Berkeley System Distribution Bloc BSS, început de simbol Semiconductor de oxid de metal complementar CMOS COFF Common Object File Format Nivelul actual de privilegii CPL Tub cu raze catodice CRT DAC Digital to Analog Converter Kit de dezvoltare a driverelor DDK DLL Biblioteca conectată dinamic DMA Acces direct la memorie DOS Disk Operating System DPL Descriptor Nivel de privilegiu Interfață pentru aplicații de nivel scăzut Întârziere pentru generarea adresei Specificare alternativă de întrerupere a multiplexorului Interfață între aplicație și program Cod standard american pentru schimbul de informații Telefon și telegraf american (compania care deținea marca comercială UNIX) Binar Decimal Sistem de bază de intrare/ieșire Cifră binară Bloc de parametri BIOS (pentru dispozitive bloc) Mod real mare (la fel ca modul ireal) Unul dintre principalele tipuri de sisteme UNIX Secțiune de program care conține date neinițializate Perechi complementare de oxid de metal Format de fișier obiect general Nivel de privilegiu curent Tub cu raze catodice Convertor digital-analogic Kit de creare a driverului Bibliotecă de legături dinamice Descriptor de sistem de operare pe disc de acces direct la memorie Nivel de privilegiu IIII Ț CPU Assembler pentru DOS, Windows și UNIX Interfață în mod protejat DPMI DOS Procesor de semnal digital DSP Zona de transfer de disc DTA ELF executabil și format de legătură Specificații de memorie extinsă EMS Bloc program de execuție EPB Tabelul de alocare a fișierelor FAT Registrul de control FCR FIFO FIFO First In First Out Mbdulație de frecvență FM Unitate FPU cu virgulă flotantă GDT Global Descriptor Table HCI Human Computer Interfață Zona de memorie înaltă HMA IBM International Business Machines Cuvânt de control al inițializării ICW Electronică de unitate integrată IDE Tabel de descriptori de întrerupere IDT IER Registrul de activare a întreruperii Control de intrare/ieșire IOCTL Nivel de privilegii de intrare/ieșire IOPL Solicitare de întrerupere IRQ Protocolul de partajare a întreruperii ISP Registrul de control al liniei LCR Tabel de descriptor local LDT LE Linear Executable Nume lung de fișier LFN Registrul de stare a liniei LSR Interfață pentru modul protejat în DOS Procesor pentru sunet digitizat pe plăcile de sunet Zona de transfer de date pe disc (în DOS) Formatul fișierelor executabile și care pot fi conectate Specificație suplimentară de acces la memorie Bloc de informații despre programul executabil Tabel de alocare a fișierelor Registrul de control FIFO Primul intrat, primul ieșit (în coadă) Sinteza de frecventa Bloc pentru lucrul cu numere în virgulă mobilă Tabel de descriptor global Interfață între utilizator și program Zona de memorie superioară ( KB după primul megaoctet) Numele companiei Cuvânt de control al inițializării Una dintre interfețele hard diskului Tabel de gestionare a întreruperii Registrul de activare a întreruperii Control I/O Nivel de privilegii I/O Cerere de întrerupere (de la dispozitivul extern) Protocol de partajare a întreruperii Registrul de control al liniei Tabel local de descriptori Format executabil liniar Nume lung de fișier Registrul de stare a liniei Abrevieri folosite eu MASM Macro Assembler Registrul de control al modemului MCR Extensie multimedia MMX Registrul de stat al modemului MSR Registrul specific al mașinii MSR NE New Executable Extensia procesorului numeric NPX Cuvânt de control al operațiunii OCW PE portabil executabil Testul Seif POST Pornire Prefixul segmentului programului PSP Registrul tampon al receptorului RBR Solicitare RFC pentru comentarii RFM Real Flat Mode Ceas în timp real RTC Nivelul de privilegiu al solicitantului RPL Notație poloneză inversă RPN Interfață SCSI Small Computer System SUN Stanford University Networks SVGA Super VGA TASM Turbo Assembler Transmițător THR Holding Register TSR Încetați și rămâneți rezident Segmentul de stare a sarcinii TSS Bloc de memorie superior UMB Extensie VBE VESA BIOS Interfața programului de control virtual VCPI Standard VESA Video Electronics Asociere Microsoft Assembler Modem Control Register Multimedia Extension Modem Status Register Machine Specific Register New Executable Format Floating Point Extension Control Word (pentru controler de întrerupere) Format executabil portabil Autotest la pornire Prefixul segmentului de program Registrul tampon de primire Solicitare de comentariu (forma de publicare a documentelor pe Internet, inclusiv standarde) Mod real plat (la fel ca modul non-real) Ceas în timp real Nivel de privilegiu solicitat Notație poloneză inversă (pentru expresii aritmetice) Una dintre interfețele hard diskului Numele companiei Orice adaptor video capabil de moduri mai mari de ore Asamblerul lui Borland Registrul de stocare al transmițătorului Termină și rămâne rezident Segment de stare a sarcinii Bloc de memorie mare (între limitele K și M) Specificație VESA pentru extinderea BIOS Una dintre interfețele pentru modul protejat pentru DOS Asociația pentru standardele video electronice II III II Asamblator pentru DOS, Windows și UNIX VGA Video Graphics Array VxD Virtual X Dispozitiv / Asamblator WASM Watcom Specificația de memorie extinsă XMS Tip adaptor video de bază Virtual Device X (nume comun pentru driverele virtuale în Windows ) Specificație de acces la memorie extinsă Watcom Assembler Glosar ȘI O înregistrare de activare este o zonă a stivei care este umplută atunci când este apelată o procedură Assembler (limbaj de asamblare) - un limbaj de programare de nivel scăzut Assembler (assembler) - un compilator din limbajul de asamblare B Byte (byte) - tip de date, având o dimensiune de biți; unitate minimă de memorie adresabilă Bit (bit) - unitatea minimă de informație LA Program pop-up - un program rezident care este activat prin apăsarea unei anumite taste „fierbinte” G Hotkey - O tastă sau o combinație de taste folosită nu pentru a introduce caractere, ci pentru a invoca programe și acțiuni similare neobișnuite d Cuvânt dublu (cuvânt dublu) - tip de date având o dimensiune de de biți Descriptor - o structură de opt octeți stocată într-unul dintre tabelele GDT, LDT sau IDT și care descrie un segment sau gateway O directivă este o instrucțiune de asamblare care nu corespunde instrucțiunilor procesorului Driver (driver) - un program utilitar care acționează ca intermediar între sistemul de operare și un dispozitiv extern O sarcină este un program, modul sau altă bucată de cod de program care poate fi pornit, executat, amânat și încheiat Protected mode (protected mode) - modul procesorului în care funcționează mecanismele de protecție, adresarea segmentelor cu descriptori și selectoare și adresarea de paginare Și Identificator (mâner sau identificator) - un număr (dacă mâner) sau o variabilă de alt tip utilizată pentru a identifica o anumită resursă ■■IUI Asamblator pentru DOS, Windows și UNIX O excepție este un eveniment la care execuția programului este încheiată și controlul este transferat către handlerul de excepții La Cod (cod) - partea executabilă a programului (un program normal este format din cod, date și o stivă) Un compilator este un program care convertește textul scris într-un limbaj de programare care poate fi citit de om într-un fișier executabil Pipeline (pipe) - o secvență de blocuri de procesor care este utilizată la executarea unei instrucțiuni Convenție (convenție) - un acord privind transferul de parametri între proceduri O mașină cu stări finite este un program care poate comuta între diferite stări și poate efectua diferite acțiuni în diferite stări Cache (cache) - memorie rapidă folosită pentru a tampona accesul la memoria principală L Limită (limită) - câmp descriptor (egal cu dimensiunea segmentului minus ) Adresă liniară (adresă liniară) - adresa obținută prin adăugarea decalajului și a bazei segmentului O capcană este o excepție care apare după comanda care a numit-o M O etichetă (labei) este un identificator asociat unei adrese dintr-un program n Un fir este un proces ale cărui date și cod sunt identice cu cele ale altor procese Modul Ireal - modul real cu limite de segmente de GB O Operand (operand) - un parametru transmis comenzii procesorului Descriptorul media este un octet folosit de DOS pentru a identifica tipul de media (de obicei nu este utilizat) O anulare este o excepție care are loc în mod asincron Evaluarea leneșă este o evaluare care se efectuează numai dacă rezultatul ei este efectiv cerut Coada de preluare este un buffer din care sunt trimise comenzi pentru decriptare și execuție O eroare este o excepție care apare înainte de comanda care a numit-o Glosar IIIIII P Pixel (pihei) - elementul minim al unei imagini raster Reintrancy - capacitatea de a începe o procedură de la un handler de întrerupere care a întrerupt executarea aceleiași proceduri Un segment conform este un segment căruia îi puteți transfera controlul către programe cu un nivel mai scăzut de privilegii Întrerupere - un semnal de la un dispozitiv extern care întrerupe execuția programului curent și transferă controlul unui program special de gestionare (vezi capcană) R Derularea buclei este transformarea buclelor care rulează de un număr cunoscut de ori într-o bucată liniară de cod Mod real (mod real) - un mod în care procesorul se comportă identic cu - adresându-se nu mai mult de un megaoctet de memorie, dimensiunea tuturor segmentelor este limitată și egală cu KB, doar modul de biți Un program rezident este un program care rămâne în memorie după ce controlul revine la DOS DIN Segment - un element de adresare a segmentului în memorie sau o secțiune a unui program DOS/Windows Secțiune (secțiune) - secțiune a programului pentru UNIX Un selector este un număr stocat într-un registru de segment Scan-code (scan-code) - orice cod trimis de tastatură Cuvântul (cuvântul) este un tip de date având o dimensiune de biți Offset - adresa relativă numărată de la începutul segmentului Stack frame - zona stivei ocupată de parametrii procedurii, o înregistrare de activare și variabile locale sau numai variabile locale Adresarea paginii (paginare) - un mecanism de adresare în care spațiul de adrese liniar este împărțit în pagini, care pot fi situate în diferite zone ale memoriei sau deloc t Jump table - o serie de adrese ale procedurilor pentru trecerea indirectă la o procedură cu un număr cunoscut w Gateway (gate) - o structură de date care permite transferul de control între diferite niveluri de privilegii într-un mod protejat Index alfabetic Adresarea indirectă cu scalare direct bazat cu indexare cu deplasare prin offset formă lungă drept registru Înregistrări de activare afișare cadru stivă Algoritmi pentru afișarea unui număr hexazecimal generatoare de numere aleatoare generare de flăcări BCD dezambalat în ASCII conversii cifre la codul ASCII hexazecimal până la zecimal cerc de desen linie dreaptă sortare Directive în limbajul de asamblare macro-uri etichete modele de memorie instrucțiuni avantaje și dezavantaje proceduri pseudo-instrucțiuni segmente structura programului asamblare condiționată Atribut simbol B Octetul Octeți de stare a tastaturii bitul Blocuri Informații VBE Parametri PSP Parametri fișiere executabile Blocurile rep în UNIX Extindere tampon tastatură cu driver LA Intrare de la dispozitivul de intrare standard de la tastatura de la mouse-ul memorie video în modurile SVGA în modurile grafice în modul text Moduri video SVGA VGA Memoria virtuală Întreruperi virtuale (în V ) Proceduri imbricate Timpul de execuție al micro-operațiilor Concluzie la stdout la ecran în modurile VGA în modul text , Tehnica de calcul virgulă mobilă cu precizie crescută punctul fix G Generatoare de numere aleatoare scădere congruente ШІ III Index alfabetic D Data și ora Descriptori segment de date sau cod Joystick Dialoguri Difuzor Biblioteci dinamice Directive de asamblare în DOS/Windows în UNIX Directoare Șoferi VxD bloc caracterul Finalizarea programului Problema Protecția memoriei Protecția paginii Mod protejat Adresare Model de memorie Selectoare DPMI VCPI Sunet fără programare DMA cu programare DMA Plăci de sunet Și Identificarea procesorului Inițializarea controlerului de întrerupere Interfață DPMI apelarea întreruperi gestionarea întreruperilor operațiuni de gestionare trecerea controlului între moduri managementul memoriei VSPI Excepții FPU SSE Mod real cod de eroare listă și funcții Executabile COFF (în UNIX) COM (în DOS) DLL (în Windows /NT) ELF EXE (în DOS) PE (în Windows /NT) SYS (în DOS) VxD (a Windows ) La Clusterul Codificări Coduri de comandă Parametri de comandă Echipe AAA AAD AAM AAS ADC d Hr ADDPS ADDSS ȘI ANDNPS ANDPS ARPL LEGAT BSF BSR BSWAP BT VTS BTR bts Apelați la CBW CDQ CLC CLD CLI CLTS CMC CMOVcc CMP CMPPS CMPS CMPSS CMPXCHG CMPXVHG B CPUID COMISS CVT* j I j III Asamblator pentru DOS, Windows și UNIX CWD CWDE DAA DAS DEC DIV DIVPS DIVSS EMMS ENTER F XM FABS FADDP FBLD FBSTP FCHS FCLEX DIV FCOM FCOM FCLEX FCOM FCOM FCOM FCOM FCOM FDIVR FD VRP FFREE FIADD FICOM FICOMP FIDIV FIDIVR FILD FIMUL FINCSTP FINIT FIST FISTULP FISUB FISUBR FISUBR FL FMULP FNCLEX FNINIT FNOP FNSAV FNSTCW FNSTENV FNSTSW FPATAN FPREM FPREM FPTAN FRNDINT FRSTOR FSAVE FSCALA FSIN FSINCOS FSQRT FST FSTCW FSTENV FSTP FSTSW FSUB FSUBP FSUBR FSURP FTST FUCOM FUCOMI FUCOMIP FUCOMP FUCOMPP FWAIT FXAM FXCH FXRSTOR , FXSAVE , FXTRACT FYL X FYL XP HLT IDIV IMUL ÎN I N C INS INT INT IN INVD INVLPG IRET JCC JCXZ JECXZ Index alfabetic II I JMP SAU LAHF ORPS LAR OUT LDMXCSR IEȘIRI LDS PACHET* LEA PADD* LASĂ PAND LES PANDN LFS PAVGB LGDT PAVGW LGS PCMP' LIDT PEXTRW LLDT PINRW LMSW PMADDDWD BLOCARE PMAX SW LODS PMAXUB LOOP PMINSW LOOPE PMINUB LOOPNE PMOVMSKB LOOPNZ PMUL* LOOPZ PMULHUW LSL POP LSS POR LTR POPA MASKMOVQ POPF MAXPS PREFETCH* MAXSS PSADBW MINPS PSLL* MINSS PSRA* MOLPS PSRL* MOLSS PSUB* MOV PUNPCK* MOVAPS PUSH MOVD PUSHA MOVHLPS PUSHF MOVHPS PXOR MOVLHPS RCL MOVLPS RCPPS MOVMSKPS RCPSS MOVNTPS RCR MOVNTQ RDMSR MOVQ RDPMC MOVS RDTSC MOVSS REP MOVSX REPE MOVUPS REPNE MOVZX REPNZ MUL REPZ MULPS RET MULSS RETF NEG RETN NOP ROL NU ROR bob i I eu a Asamblator pentru DOS, Windows și UNIX RSM RSQRTPS RSQRTSS SAHF SAL SALC SAR SBB SCAS SETcc SGDT SHL SHLD SHR SHRD SHUFPS SHUFW SIDT SLDT SMSW SQRTPS SQRTSS STC STD STI STMXCSR STOS STR SUB SUBPS SUBSS SYSENTER SYSEXIT TEST UCOMISS UD UNPCHPS UNPCKLPS VERR VERW Așteptați WBINVD WRMSR XADD XCHG XLAT XOR XORPS identificarea procesorului Extensie AMD D Compilare' în format COFF la fișierul COM în DLL la formatul ELF Fișierul WEXE în UNIX drivere pentru DOS aplicație grafică cu resurse aplicație de consolă folosind extensii DOS Conducte de execuție a comenzilor Convenții de transmitere a parametrilor Convenția C Convenția PASCAL mixt Aparatele de stat Aplicații de consolă Controlor DMA întrerupe L Linear Frame Buffer (LFB) Linia A Operații logice m Definiții macro în UNIX Micro-operații Octet scăzut LSB Multitasking în DOS Modele de memorie Modemuri Mouse n Saturație Modul Ireal Multitasking cu thread-uri O Notație poloneză inversă Mediu DOS Operanzi ' Operatori în asamblatoarele AT&T OS DOS Linux, FreeBSD, Solaris Windows /NT Optimizare programe nivel înalt nivel mediu nivel scăzut ' cicluri Index alfabetic Organizare întârzieri memorie modele de memorie endian segmente stiva Registre de depanare P Paleta VGA Memorie XMS alocarea definiție maximă a blocului ediția Trecerea parametrilor în blocul de parametri în variabile globale în fluxul de cod în registrele stivuite în limbi de nivel înalt evaluare leneșă după valoarea returnată după valoarea după nume prin rezultatul linkul Comutare băncile sarcini Variabile de mediu Reintrare în BIOS în DOS Programe semirezidente Porturi VGA DAC Controler VGA CRT Sincronizator VGA tastatura paralel serial Predicția tranziției Întreruperile în DPMI în modul protejat inițializarea controlerului gestionarea întreruperii de la dispozitive externe activare și dezactivare Prefixul segmentului de program (PSP) Prefixe BLOCAREA REP REPE REPNE REPNZ REPZ alte prefixe Comă privilegiată-d= ' Proceduri Procesoare Pentium Pro și Pentium II - Pentium și Pentium MMX Pseudo-comenzi pentru a defini ds s CE în UNIX R Extensii de paginare Coduri ASCII extinse Extensii DOS Modul real Registrele CRx DRx MSR scop general date FPU MMX segmentul cuvinte de stare FPU Cuvinte de control FPU managementul memoriei Steaguri CPU Moduri X Moduri procesor RFM/BFM V ■ protejat ireal real Programe rezidente descărcare din memorie fără PSP întrerupere multiplexer pasiv și activ reintrare programe semirezidente caietul de sarcini AMIS DIN Segmentul de stare a sarcinii Adresarea segmentului în mod protejat Segmente ] l JI Asamblator pentru DOS, Windows și UNIX Sectorul Secțiunile Selectoare , Caractere ASCII , Notaţie binar hexazecimal funcțiile sistemului libc UNIX win Temporizator de sistem la nivelul BIOS la nivelul porturilor de intrare-ieșire Scanare coduri Viteza de executare a comenzii Word Mesaje (pe Windows) Triere rapid bubble alegere Octet înalt MSB Legături statice cadru stiva Adresarea paginii Protecție pentru adresarea paginii Extensii Pentium Pro T Sari masa Temporizator la nivelul BIOS la nivelul porturilor I/O Tipuri de date Caractere ASCII MMX octeți împachetati cuvinte duble cuvinte cuvinte patru octetul bit numere reale lung scurt extins cazuri speciale bcd cuvânt dublu Managementul sarcinilor Registre de control Caractere de control ASCII Nivel de imbricare Salturi condiționate Dispozitive Adaptoare video VGA joystick difuzor scrierea pe un dispozitiv plăci de sunet tastaturi controlere DMA întrerupe cronometrul de sistem F Fișiere record ID descoperire căutare căutare de nume lung creație îndepărtarea Steaguri sistemul Indicatori de stare FPU CPU Funcții în sistemul de asamblare libc win în UNIX Cicluri PENTRU LOOP/ENDLOOP REPETĂ/PÂNĂ LA ÎN patru Ceas în timp real la nivelul BIOS la nivelul porturilor I/O Numere punctul fix virgulă mobilă Gânduri l <ADEMIA L OF DOGESTEV URSS GÂNDIRE TÂNĂ LUMEA ÎN CINCI VOLUMUL X echipa editorială MF OVSYANNIKOV (EDITOR SEF) I GERSHKOVICH, M L LIVSHITS, V S m Nikiforov A Ya REINGARD, P M SYSOEV, V P ȘHESGAKOV EDITURA ACADEMIEI URSS Institutul de Cercetare de TEORIA SI ISTORIA ARTELOR FRATICE ISTORIA STEGIKA apoi M ÎNTÂI ANTICHITATE EVUL MEDIU RENAŞTERE Editor-compilator al volumului V P SHESTAKOV EDITORIAL Antologia despre istoria esteticii, publicată de Institutul de Cercetări de Teorie și Istorie a Artelor Plastice al Academiei de Arte din URSS, este formată din cinci volume Primul volum acoperă dezvoltarea gândirii estetice din antichitate până în Renaștere Volumele al doilea, al treilea și al patrulea conțin materiale care reflectă învățăturile estetice ale secolelor al XVII-lea, al XVIII-lea și al XIX-lea, volumul al cincilea este dedicat dezvoltării esteticii marxist-leniniste Primul volum prezintă învățăturile estetice ale Greciei și elenismului clasic, gândirea estetică medievală a Europei de Vest, Bizanț, India, China și Rusia, istoria gândirii estetice a Renașterii Articolul introductiv la primul volum conține o descriere generală a dezvoltării gândirii estetice în aceste etape ale dezvoltării sale În plus, articolele de recenzie sunt trimise unor perioade individuale, gânditori, precum și unor monumente Monumentele gândirii estetice, de regulă, nu sunt date în întregime, ci în fragmente separate, cele mai semnificative Totodată, sunt date integral câteva texte puțin cunoscute cititorului sovietic Toate secțiunile primului volum conțin o cantitate semnificativă de materiale publicate în limba rusă pentru prima dată (aceste noi traduceri sunt indicate în text prin două asteriscuri lângă numele traducătorului) Pentru prima dată materialele publicate despre estetica Indiei, Chinei și Rusiei sunt însoțite de notele necesare Textele publicate anterior sunt reproduse după cele mai complete și moderne ediții Fiecare secțiune a cărții este însoțită de o bibliografie care familiarizează cititorul cu principalele ediții ale monumentelor gândirii estetice și ale literaturii de referință în rusă și limbi străine Următoarele volume ale acestei ediții sunt întocmite pe același principiu Subiectele și indicii nominali vor fi publicate în Volumul IV (până la primele patru volume) și Volumul V (Estetica marxist-leninistă) V E D E N I educația estetică a poporului face parte din educația comunistă, al cărei scop cel mai înalt este dezvoltarea integrală și armonioasă a personalității umane Importanța sa crește mai ales în perioada construcției extinse a comunismului Trebuie să fie un mijloc puternic de îmbogățire spirituală pentru constructorul comunismului, al dezvoltării sale morale și politice Sarcinile în domeniul educației estetice sunt clar definite în Programul Partidului Comunist al Uniunii Sovietice, adoptat la Congresul XXII: „Partidul se va ocupa neobosit de despre înflorirea literaturii, artei, culturii, despre crearea tuturor condițiilor pentru manifestarea cât mai deplină a abilităților personale ale fiecărei persoane, despre educația estetică a tuturor oamenilor muncii, formarea unor înalte gusturi artistice și abilități culturale în rândul oamenilor Principiul artistic va spiritualiza și mai mult munca, va decora viața și va înnobila o persoană În acest sens, este de mare importanță dezvoltarea teoretică a problemelor fundamentale ale esteticii marxist-leniniste: spiritul de partid și arta populară, internațională și națională în artă, tradiția și inovația în creativitatea artistică, fundamentele filozofice ale realismului socialist, rolul și semnificația artei în construcția comunistă, formă și conținut, în artă, meșteșuguri etc Cu toate acestea, o dezvoltare profundă a problemelor estetice este imposibilă fără o analiză critică a istoriei gândirii estetice, dezvoltarea tuturor celor mai bune și avansate, acumulate de-a lungul secolele dezvoltării sale, fără o luptă activă împotriva conceptelor idealiste și metafizice de artă și creativitate artistică În legătură cu sarcinile luptei ideologice în stadiul actual și, în special, cu expunerea ideologiștilor burghezi și ai lor ai revizioniștilor, care încearcă să „infirme” și „corecteze” marxismul, estetica marxist-leninistă, dezvoltarea dintre aceste probleme este de mare relevanță V I Lenin a subliniat în mod repetat importanța enormă a stăpânirii celor mai valoroase realizări culturale din epocile anterioare În proiectul de rezoluție „Despre cultura proletariană”, el a scris: „Marxismul și-a câștigat semnificația sa istorică mondială ca ideologie a proletariatului revoluționar prin faptul că marxismul nu a renunțat în niciun caz la cele mai valoroase câștiguri ale erei burgheze, dar, dimpotrivă, a asimilat și procesat tot ceea ce era valoros în peste două mii de ani de dezvoltare a gândirii și culturii umane Numai munca ulterioară pe această bază și în aceeași direcție, inspirată de experiența practică a dictaturii proletariatului ca ultima sa luptă împotriva oricărei exploatări, poate fi recunoscută ca dezvoltarea unei culturi cu adevărat proletare Filosofii, teoreticienii și istoricii de artă sovietici au făcut o treabă grozavă studiind moștenirea estetică a trecutului Principalele etape din istoria gândirii estetice au fost deja analizate în detaliu: antichitatea, Renașterea, clasicismul, Iluminismul, estetica clasică germană, estetica democraților revoluționari ruși, precum și estetica democratică revoluționară din alte țări; publicat și comentat de mulți * Programul PCUS M , , p ** V I Lenin Opere, vol , p cele mai importante monumente ale gândirii estetice mondiale; au fost criticate principalele tendințe ale esteticii burgheze și revizioniste contemporane; Moștenirea estetică a lui Marx, Engels, Lenin a fost studiată în detaliu ca fiind cea mai înaltă etapă din istoria dezvoltării gândirii estetice mondiale În prezent, au fost întreprinse studii pe scară largă asupra gândirii estetice a țărilor slave, a țărilor din Est, Asia Centrală și țările arabe Acum există toate premisele pentru a scrie o lucrare monumentală despre istoria gândirii estetice mondiale Această sarcină serioasă, desigur, poate fi rezolvată doar ca rezultat al eforturilor colective Încercările de a scrie o istorie sistematică a gândirii estetice au fost făcute în mod repetat de mulți savanți burghezi: R Zimmermann, M Schasler în Germania, B Bosanquet, C Gilbert și H Kuhn în Anglia, B Croce în Italia și Bayer în Franta Aceste încercări au eșuat de obicei Adevărat, în studiile burgheze despre istoria esteticii se pot găsi uneori materiale faptice valoroase și considerații interesante cu privire la anumite chestiuni În general, aceste lucrări sunt insuportabile din cauza falsității principiilor metodologice care le stau la baza Principalul defect al metodologiei studiilor burgheze în istoria esteticii este interpretarea idealistă a vieții sociale și a tuturor fenomenelor și aspectelor ei Înțelegerea idealistă a istoriei îi determină pe autorii burghezi să considere gândirea estetică ca pur imanentă, izolată de toate aspectele activității sociale umane Savanții burghezi ignoră legătura dintre teoriile estetice și baza materială a societății și a luptei de clasă Separând gândirea estetică de procesul integral al istoriei, savanții burghezi își blochează astfel drumul către o înțelegere adevărată a originii și dezvoltării sale Istoricul esteticii este obligat să explice de ce apar anumite teorii estetice într-o anumită perioadă, ce explică ascensiunea și căderea gândirii estetice, de ce unele concepte estetice sunt înlocuite de altele și ce provoacă lupta în vederi asupra principalelor probleme estetice, si in final, care ar trebui sa fie criteriul de evaluare a acestor concepte Toate aceste întrebări pentru istoricul gândirii estetice, care se află pe poziții idealiste, rămân un mister în spatele a șapte peceți Prin urmare, nu este nimic surprinzător în faptul că savanții burghezi ar putea oferi doar cele mai superficiale descrieri ale esteticii individuale nouă teorii sau o colecție de eseuri împrăștiate care reproduc doar istoria externă a învățăturilor estetice Sarcina istoriei științifice a esteticii este de a dezvălui esența interioară și modelul apariției și dezvoltării gândirii estetice O astfel de sarcină poate fi rezolvată doar pe baza metodologiei marxiste Din punctul de vedere al marxismului, ideile estetice sunt fața unui proces istoric holistic, în care dezvoltarea forțelor productive în interacțiunea cu relațiile de producție joacă un rol decisiv Ideile estetice, ca orice alte idei, sunt o reflectare a procesului real al vieții oamenilor Prin urmare, sursa apariției și dezvoltării lor, înflorirea și declinul lor, trebuie căutată nu în ideile în sine, așa cum fac idealiștii, ci în condițiile materiale ale societății, în natura și caracteristicile luptei de clasă în fiecare etapă individuală a dezvoltarea socială într-o anumită țară Ideile estetice sunt în interacțiune vie cu alte elemente ale suprastructurii - politică, morală, artă Prin urmare, dezvoltarea acestora din urmă trebuie luată în considerare pentru analiza anumitor concepte estetice Teoriile estetice nu sunt un produs pasiv al bazei economice Au independență relativă, capacitatea de a influența înapoi existența socială a oamenilor care i-au născut Fiecare teoretician al artei, orice estetician exprimă direct sau indirect interesele unei clase sau alteia, adică ia o anumită poziție ideologică și luptă pentru realizarea idealurilor unei anumite clase Estetica petrecerii Așa a fost acum mai bine de două mii de ani și așa rămâne până în zilele noastre Este exact ceea ce studenții burghezi ai istoriei esteticii nu vor să vadă Între timp, întreaga istorie veche de secole a gândirii estetice nu este altceva decât istoria originii, formării, dezvoltării principalelor tendințe din estetică - materialism și idealism, istoria luptei dintre învățăturile estetice materialiste și teoriile idealiste Această luptă între materialism și idealism în estetică este deja relevată în atitudinea critică a lui Heraclit față de ideile religioase și mitologice; se reflectă în controversa dintre Aristotel și Platon, în lupta esteticii Renașterii împotriva conceptelor scolastice medievale, în discursurile pasionate ale lui Diderot și Lessing împotriva esteticii clasicismului secolului al XVII-lea, în conflictele ascuțite dintre Goethe, pe pe de o parte, Schelling și romanticii, cu alta Astfel de exemple pot fi multiplicate Trebuie să fii orb ca să nu vezi în această luptă, o reflectare a luptei claselor, grupurilor sociale, expresia diferitelor concepte morale și politice A urmări toate vicisitudinile acestei lupte nu înseamnă altceva decât a recrea o istorie adevărată, obiectivă, a dezvoltării gândirii estetice O încercare de a scrie istoria esteticii în afara luptei dintre materialism și idealism, care este o expresie a luptei de clasă, duce inevitabil la o denaturare a imaginii dezvoltării gândirii estetice mondiale Teoreticienii idealiști burghezi nu înțeleg și nu doresc să înțeleagă natura de clasă socială a teoriilor estetice Având în vedere acest lucru, ei nu au reușit să periodizeze corect istoria esteticii Metodologia marxistă oferă cheia rezolvării acestei probleme Într-adevăr, dacă pornim de la faptul că ideile estetice fac parte din suprastructură, atunci, în consecință, istoria esteticii trebuie privită din punctul de vedere al schimbării formațiunilor socio-istorice Desigur, aici este necesar să se țină cont de relativa independență a gândirii estetice, de logica internă a dezvoltării acesteia, de legătura ei cu arta, filosofia etc În această lumină, problema periodizării științifice este destul de rezolvabilă și are deja rezolvată în istoria marxist-leninistă a esteticii Care este opinia autorilor burghezi în acest sens? Ca exemplu, putem lua „Istoria esteticii” de Katharina Gilbert și Helmut Kuhn, publicată în limba rusă în Cărții îi lipsește orice principiu clar al periodizării Gilbert și Kuhn se străduiesc să organizeze cumva materialul istoric, dar nu au un criteriu obiectiv de grupare a materialului Rezultatul este un amestec bizar de concepte și concepte care sunt complet eterogene sau au asemănări destul de îndepărtate Alți autori burghezi, precum Zimmermann, Bozanket, Croce și alții, au o situație similară cu periodizarea: nici ea nu reflectă tabloul istoric real Idealismul istoricilor burghezi ai esteticii se manifestă în atitudinea lor disprețuitoare față de teoriile democratice și materialiste ale esteticii Este zadarnic să cauți în aceste lucrări o expunere a concepțiilor estetice ale lui Helvetius, Forster, Buchner, Heinrich Heine, de exemplu Atitudinea părtinitoare față de materialism în estetică este exprimată de autorii burghezi nu numai în desconsiderare directă față de estetica materialistă: adesea teoriile lor sunt prezentate în așa fel încât nu rămâne nimic din materialism Părerile lui Aristotel, Diderot, Goethe și mulți alți gânditori de seamă sunt supuse unei astfel de „operații” Dar când vine vorba de estetica idealistă, cercetătorii burghezi nu se zgârcesc cu laudele P și acordă-le un loc disproporționat de mare în scrierile lor Aici, de altfel, se dezvăluie adevărata natură a obiectivismului burghez Cu cuvinte, oamenii de știință burghezi sunt pentru o cercetare imparțială din punct de vedere științific Îi poți auzi adesea vorbind despre modul în care știința ar trebui, spun ei, să se ferească de politică și părtinire de partid În realitate însă, o astfel de poziție obiectivistă servește doar ca acoperire pentru partizanatul burghez Prejudecata de clasă a autorilor burghezi este evidentă mai ales în atitudinea lor față de estetica marxist-leninistă Marxismul, după cum știm, există de mai bine de o sută de ani De la începuturile sale, a devenit o armă ideologică puternică pentru transformarea revoluționară a societății Ea a ocupat de mult timp un loc decisiv în viața spirituală a omenirii Marxismul este în prezent o forță atractivă pentru toți oamenii progresiști ai lumii Marxismul, ca sistem armonios și integral de vederi filozofice, economice și socio-politice, conține un sistem științific de estetică Estetica marxistă a fost dezvoltată în lucrările lui Marx, Engels, Lenin și în lucrările asociaților, studenților și adepților lor - F Mehring, P Lafargue, G Plekhanov, A Lunacharsky, V Vorovsky Istoricii burghezi ai esteticii fie tăcesc existența esteticii marxiste, fie o falsifică, fie o supun atacurilor aprige Despre ce fel de imparțialitate științifică pot vorbi oamenii de știință burghezi? Spiritul de partid burghez a găsit o expresie deosebită în așa-numita concepție eurocentrică, a cărei esență reacționară a fost dezvăluită în detaliu de savanții marxisti Dacă credem autorii burghezi, atunci gândirea estetică a omenirii este un produs al activității numai popoarelor vest-europene, prin urmare, în lucrările despre istoria generală a esteticii scrise de autori burghezi, nu vom găsi, de exemplu, o reflectare a gândirea estetică a Indiei, Chinei, Rusiei, Poloniei, Bulgariei și a altor țări Reversul acestui punct de vedere eurocentric, naționalist este nihilismul național De fapt, gândirea estetică este strâns legată de dezvoltarea artei și literaturii unui anumit popor, iar acestea din urmă, la rândul lor, reflectă căile specifice ale dezvoltării istorice a unui anumit popor Cu toate acestea, în cărțile savanților burghezi nu există o analiză a trăsăturilor distinctive ale dezvoltării esteticii în profunzimea culturii unui anumit popor Inconsecvența metodologică menționată mai sus a lucrărilor burgheze despre istoria esteticii sărăcește total conținutul acestora și mărturisește inconsecvența istoriei burgheze a esteticii ca știință Din punctul de vedere al metodologiei marxist-leniniste, istoria esteticii este un proces natural al vieții ideologice în curs de dezvoltare, în care diverse învățături estetice sunt interconectate, interdependente, în continuă schimbare în cursul luptei tendințelor și direcțiilor opuse Această luptă duce la schimbări calitative ale esteticii, care culminează cu o răsturnare revoluționară, și apare următoarea etapă a dezvoltării sale Compilatorii acestei lucrări în selecția și aranjarea materialului, în introducerea volumelor și articolele introductive la secțiuni individuale de volume, s-au ghidat de acest principiu fundamental al cunoașterii marxist-leniniste a istoriei esteticii și a legilor sale Aici a fost important să arătăm natura firească a dezvoltării ideilor estetice, legătura lor cu modul de producție, cu lupta de clasă într-o societate de clasă, ținând cont atât de condițiile naționale, cât și internaționale pentru dezvoltarea lor Deoarece dezvoltarea ideilor estetice, ca dezvoltarea ideilor filosofice, juridice, etice și de altă natură, are o relativă independență, s-a luat în considerare și acest aspect Ideile estetice sunt legate de dezvoltarea artei și a altor forme de cunoaștere estetică și activitatea umană: filozofie, idei politice etc Acest fapt este de obicei ignorat de știința burgheză Compilatorii acestei antologii, dimpotrivă, au căutat să identifice și să analizeze această legătură Sarcina de a depăși teoria reacționară a „eurocentrismului” în istoria esteticii a fost și ea extrem de importantă În acest sens, au fost întreprinse lucrări semnificative pentru identificarea și analizarea materialelor privind istoria dezvoltării ideilor estetice în China, India, Rusia și alte țări Publicația va acorda o mare atenție și istoriei gândirii estetice marxist-leniniste Cititorul în cinci volume oferit cititorului este prima încercare de a crea o astfel de publicație Primul volum al acestei publicații acoperă perioada istoriei esteticii de la antichitate până la Renaștere, inclusiv Scopul introducerii în primul volum este de a contura doar contururile cele mai generale ale dezvoltării gândirii estetice în perioada amintită mai sus Această imagine este completată în articole introductive la secțiuni individuale ale cărții, autori și monumente ale gândirii estetice i * * * Ideile și ideile estetice apar chiar și în societățile de sclavi din Orientul Antic (Babilon, Egipt) Cu toate acestea, ideile estetice au primit cea mai detaliată dezvoltare, în principal în societatea antică de sclavi Referitor la moștenirea filozofică a vechilor greci, Engels spune că suntem siliți „din nou și din nou în filosofie, ca și în multe alte domenii, la realizările acelui popor mic, al cărui talent și activitate universală i-au asigurat un loc în istorie al dezvoltării omenirii, pe care nu o pot ocupa, niciun alt popor nu poate pretinde în diversele forme ale filozofiei grecești, există deja în embrion, în curs de apariție, aproape toate tipurile ulterioare de viziuni asupra lumii”* Ceea ce a spus Engels despre filosofia greacă antică se aplică și esteticii grecilor antici Grecii antici au dat exemple strălucitoare de abordare dialectică a fenomenelor realității, inclusiv a fenomenelor artei, deși dialectica lor este încă caracterizată de simplitate naivă și este rezultatul unei contemplări directe a naturii și a vieții sociale Gânditorii antici au formulat principalele probleme de estetică și au dezvoltat cele mai importante concepte și termeni estetici Concepte precum imitație, inspirație, frumusețe, tragic, comic, ironie, măsură, armonie, ideal și multe altele au fost definite de grecii antici și au fost ulterior fixate în literatura estetică și de istoria artei Estetica antică a apărut ca o încercare de a înțelege filozofic practica artistică a timpului său Legătura sa profundă cu arta modernă și marile probleme ale epocii sale fac gândirea antică deosebit de interesantă de studiat Gândirea estetică a Greciei antice a avut o influență puternică asupra dezvoltării ulterioare a esteticii (Renașterea, clasicismul, Iluminismul etc ) Ideile antice au fost folosite ca armă spirituală în lupta împotriva feudalismului în timpul Renașterii, în timpul Marii Revoluții Franceze Influența antichității a fost deosebit de puternică în estetica clasică germană (Forster, Winckelmann, Lessing, Schiller, Goethe, Hegel și alții) Acum nu există niciun motiv pentru a idealiza democrația antică, așa cum au făcut, de exemplu, iacobinii sau filozofii germani de la sfârșitul secolului al XVIII-lea și începutul secolului al XIX-lea Cultura antică a apărut într-o societate *LA Marx şi F Engels Opere, vol , p paisprezece care a fost puternic împărțit în liberi și sclavi, unde toată viața se sprijinea pe baza exploatării muncii sclavilor, unde întreaga luptă ideologică se desfășura numai între diferite secțiuni ale „născuților liberi”, deoarece sclavii erau excluși de la participarea la viata spirituala a societatii Această împrejurare a lăsat o amprentă a limitărilor istorice asupra esteticii antice Totuși, acest lucru nu dă ideologiștilor epocii imperialismului dreptul de a falsifica sistemul democratic antic și cultura antică și să încerce să găsească în politica antică acele contradicții care sunt caracteristice societății burgheze Acest antiistoricism fundamental în evaluarea antichității este o formă specifică de apologie a capitalismului Doar fondatorii marxismului au dat o evaluare cu adevărat științifică a moștenirii culturale antice Astfel, referindu-se la arta greacă, Marx notează în primul rând că aceasta era strâns legată de mitologie, care nu era doar un arsenal, ci și pământul său „Toată mitologia”, spune Marx, „învinge, supune și modelează forțele naturii în și prin imaginație; dispare, în consecință, odată cu declanșarea dominației reale asupra acestor forțe ale naturii Condiția prealabilă pentru arta greacă este mitologia greacă, adică natura și formele sociale înseși, deja reelaborate într-un mod inconștient artistic de fantezia populară Acestea sunt lucrurile lui ” Marx leagă înflorirea artei grecești antice cu particularitățile formei sociale de viață a grecilor Marx, însă, nu se limitează să sublinieze legătura dintre artă și anumite forme de dezvoltare a vieții sociale El dezvăluie motivele pentru care arta greacă continuă să ne ofere plăcere artistică și, într-un anumit sens, păstrează semnificația normei și a modelului de neatins „Un bărbat”, spune Marx, „nu se poate transforma înapoi într-un copil fără să cadă în copilărie Dar naivitatea copilului nu-i mulțumește și nu ar trebui el însuși să se străduiască să-și reproducă adevărata natură la un nivel superior? Oare firea copilului în fiecare epocă nu reînvie propriul său caracter în adevărul ei fără pricepere? Și de ce să nu aibă copilăria societății umane, unde s-a dezvoltat cel mai frumos, un farmec etern pentru noi, ca un pas care nu se repetă niciodată?** * K Marx şi F Engels Opere, vol , p ** T și m Miturile grecilor se disting prin patosul optimist care afirmă viața Umbra lui Ahile, referindu-se la Ulise, spune: Ar fi mai bine dacă aș fi în viață, ca un zilier, lucrând la câmp, Slujind un biet plugar ca să-mi ia pâinea cea de toate zilele, decât să domnesc peste morții fără suflet de aici, morți* Mitologia greacă era de natură antropomorfă Zeii aici apar sub formă de oameni și sunt caracterizați de pasiuni și slăbiciuni umane Imaginea zeilor greci întruchipa idealurile pozitive ale grecilor Acest tip de mitologie putea servi drept material pentru sculptura, literatura și teatrul grecești Nu întâmplător, majoritatea operelor de artă grecească se bazează pe prelucrarea miturilor În momentul în care au apărut primele concepte estetice, arhitectura, sculptura și poezia atinseseră un nivel înalt de dezvoltare în Grecia antică Aceasta explică în mare măsură profunzimea dezvoltării unui număr de probleme din teoria artei, apariția unor tratate de estetică atât de semnificative precum Poetica lui Aristotel Înflorirea gândirii filozofice în Grecia, mai ales în secolele V-IV î Hr e , a contribuit și la dezvoltarea teoriei artei și a esteticii Teoreticienii proeminenți ai artei Greciei antice sunt pitagoreicii Uniunea Pitagoreică, fondată de Pitagora (secolul VI î Hr ), a prezentat o serie de astronomi, matematicieni și filozofi importanți După părerile lor, pitagoreicii sunt idealiști Potrivit acestora, numărul este esența lucrurilor și, prin urmare, cunoașterea lumii se reduce la cunoașterea numerelor care o controlează Aceste propoziții filozofice de bază au servit drept puncte de plecare pentru construcțiile lor estetice Pitagorei cred că armonia numerelor este un fel de regularitate obiectivă care operează în toate fenomenele vieții și, prin urmare, în artă Meritul pitagoreenilor este că ei au fost primii care au interpretat armonia ca o unitate a contrariilor Conceptul general de armonie a fost aplicat de ei muzicii Ei au fost primii care au prezentat ideea că originalitatea calitativă a unui ton muzical depinde de lungimea coardei care sună Pe această bază, au dezvoltat doctrina matematicii, fundamentele intervalelor muzicale Deci, au stabilit următoarele armonii muzicale: octava - : , quart - : , a cincea - : Este important de subliniat faptul că pitagoreicii caută o bază obiectivă pentru fenomenele estetice, iar această bază , dupa parerea lor * Homer Odiseea, - IIer V Jukovski şaisprezece considerate a fi cuantificabile În ciuda premiselor idealiste inițiale, pitagoreicii au ridicat corect problema fundamentelor obiective ale frumuseții În estetica pitagoreenilor, se acordă multă atenție clarificării rolului educațional al artei Potrivit lui Yamvlich, Pitagora a acordat o mare importanță muzicii ca mijloc de „vindecare a moravurilor și pasiunilor umane” Muzica, potrivit lui Pitagora, contribuie la educația morală a oamenilor Pitagorei echivalează armonia, perfecțiunea și frumusețea Cu toate acestea, pitagoreicii au înțeles armonia ca „acordul celor care nu sunt de acord”, adică au subliniat reconcilierea contrariilor Acesta este deficiența doctrinei lor despre armonie Mult mai profund, problema contradicției în raport cu estetica a fost reținută de fondatorul dialecticii, Heraclit din Efes (sfârșitul secolului al VI-lea - începutul secolului al V-lea î Hr ) Continuând tradițiile materialiste ale școlii filozofice milesiene, Heraclit ia „focul etern viu” drept începutul tuturor lucrurilor Regularitatea strictă domnește în lume și, în același timp, nu există nimic constant în ea - totul curge și se schimbă Potrivit lui Aristotel, Heraclit credea că „totul ia naștere prin luptă” Spre deosebire de pitagoreici, Heraclit se concentrează nu pe reconcilierea contrariilor, ci pe lupta lor Heraclit, ca și pitagoreenii, crede că frumosul are o bază obiectivă, dar el vede această bază nu în relațiile numerice, ca atare, ci în calitățile lucrurilor materiale, care sunt modificări ale focului Frumusețea, de altfel, este o proprietate relativă: „Cea mai frumoasă maimuță”, spune Heraclit, „este urâtă în comparație cu rasa umană”* Relativitatea frumosului este determinată de apartenența ființelor la diferite genuri Mu 